License: CC BY 4.0

1 📕 READ ME

🚧 UNPUBLISHED AND ONGOING example 🚧 The data in this script are currently unpublished, and the analysis is ongoing. Some sections may be incomplete. This script has been made publicly available to facilitate sharing with collaborators and colleagues who are interested in the progress of the project and to ensure transparency.

SUMMARY This R code estimates the relationship between oxygen consumption (MO₂) and ambient oxygen partial pressure (PO₂) in the Common Galaxias (Galaxias maculatus). It also estimates the critical partial pressure of oxygen for aerobic metabolism (Pcrit), commonly defined as the threshold below which the oxygen consumption rate can no longer be sustained. The associated article is titled “The role of osmorespiratory compromise in hypoxia tolerance of the purportedly oxyconforming teleost Galaxias maculatus.” If you are reading the HTML version of this script, click the Code button in the top right to download the .Rmd file.

AIM The article aims to test whether Galaxias maculatus can maintain oxygen consumption (MO₂) as ambient PO₂ declines and, if so, at what level it reaches the critical partial pressure of oxygen for aerobic metabolism (Pcrit).

AUTHORS Timothy D. Clark [a] Luis L. Kuchenmüller [a] Elizabeth C. Hoots [a] Maryane Gradito [a] Jake M. Martin [a,b] In no particular order

AFFILIATIONS [a] School of Life and Environmental Sciences, Deakin University, Geelong, VIC, Australia
[b] School of Biological Sciences, Monash University, Clayton, VIC, Australia

CONTRIBUTOR ROLES 🚧 To be added 🚧 Based on the Contributor Roles Taxonomy (CRediT)

DISCLAIMER I (Jake Martin) am dyslexic. I have made an effort to review the script for grammatical errors, but some will likely remain. I apologise. Please feel free to reach out using the contact details below if anything is unclear.

2 📧 Contact

Jake M. Martin

📧 Email: jake.martin@deakin.edu.au

📧 Alt Email: jake.martin.research@gmail.com

🌐 Web: jakemartin.org

🐙 GitHub: JakeMartinResearch

3 📑 Sharing/accessing and citing

  1. Licenses/restrictions placed on the data: CC-BY 4.0

  2. Link to the associated publication:
    🚧 To be added 🚧

  3. Recommended citation for this data:
    🚧 To be added 🚧

4 📦 Required packages and knit settings

A list of the required packages to run the script.

# ---- Install pacman if it's not already installed ----
if (!requireNamespace("pacman", quietly = TRUE)) install.packages("pacman")

# ---- List of required packages ----
pkgs <- c(
  # ----- Data Visualisation -----
  "ggthemes", "bayesplot", "gt", "gtsummary", "plotly", "qqplotr", "gridExtra",
  
  # ----- Tidy Data and Wrangling -----
  "tidyverse", "janitor", "readxl", "broom.mixed", "data.table", "hms", "devtools",
  "mclust",
  
  # ----- Modelling and Statistical Analysis -----
  "brms", "rstan", "marginaleffects", "performance", "emmeans",
  "tidybayes", "respirometry", "future"
)

# ---- Install and load all packages using pacman ----
suppressPackageStartupMessages(
  pacman::p_load(char = pkgs, install = TRUE)
)

#kniter seetting
knitr::opts_chunk$set(
message = FALSE,
warning = FALSE, # no warnings
cache = TRUE,# Cacheing to save time when kniting
tidy = TRUE,
fig.align = "center" 
)

5 🔧 Custom functions

Here are some custom functions used within this script.

bayes_incremental_regression_by_id(): A custom function to build Bayesian incremental regressions. It is designed to run a list of subgroup models (IDs) in parallel using 4 cores. The function uses brm() with a Gaussian error distribution.

Use:
The function accepts the following arguments:
- id_i: A grouping factor or ID used to filter the data for each regression. If none is provided, the function uses the entire dataset.
- id_name: The column name (as a character string) corresponding to the grouping factor in the data frame. If not provided, the function uses all data.
- data: The data frame containing the variables for analysis.
- predictor: The predictor variable of interest.
- response: The response variable of interest.
- seed_number: A random seed value for model reproducibility.
- save_models: A logical argument indicating whether to save the model outputs (TRUE or FALSE).
- mod_output_wd: The output directory where model .rds files should be saved (used only if save_models = TRUE).

# Instead we could also use a distributional regression approach, by
# specifically modelling the variance by DO (e.g. sigma ~ DO). Weighting may
# not be required in this case, I don't think higher density of values in a
# given space will effect Bayesian estimates like it does in frequentist
# models. See discourse https://discourse.mc-stan.org/t/weights-in-brm/4278

bayes_incremental_regression_by_id <- function(id_i, id_name, data, predictor, response,
    seed_number, save_models, mod_output_wd) {
    # Initiate an empty list to store models
    models <- list()

    # Check if id_name is missing, NULL, or blank, and assign NA if so
    if (missing(id_name) || is.null(id_name) || id_name == "") {
        id_name <- NA
    }

    # Check if id_i is missing, NULL, or blank, and assign NA if so
    if (missing(id_i) || is.null(id_i) || id_i == "") {
        id_name <- NA
    }

    # Filter data for the current ID if id_name is given as a factor or
    # character and id_i is defined
    df_i <- data %>%
        dplyr::filter(if (!is.na(id_i) && (is.factor(data[[id_name]]) || is.character(data[[id_name]]))) {
            !!rlang::sym(id_name) == id_i
        } else {
            TRUE
        })

    # Dynamically create formulas
    formula_lm_0 <- reformulate("1", response)
    formula_lm_1 <- reformulate(predictor, response)
    formula_lm_2 <- reformulate(sprintf("poly(%s, 2)", predictor), response)
    formula_lm_3 <- reformulate(sprintf("poly(%s, 3)", predictor), response)

    # Fit and store models in the list
    models[[paste0(id_i, "_lm_0")]] <- brm(bf(formula_lm_0, family = gaussian()),
        data = df_i, cores = 4, seed = seed_number, save_pars = save_pars(all = save_models),
        sample_prior = FALSE, silent = TRUE, file = paste0(mod_output_wd, "/", id_i,
            "_lm_0"))

    models[[paste0(id_i, "_lm_1")]] <- brm(bf(formula_lm_1, family = gaussian()),
        data = df_i, cores = 4, seed = seed_number, save_pars = save_pars(all = save_models),
        sample_prior = FALSE, silent = TRUE, file = paste0(mod_output_wd, "/", id_i,
            "_lm_1"))

    models[[paste0(id_i, "_lm_2")]] <- brm(bf(formula_lm_2, family = gaussian()),
        data = df_i, cores = 4, seed = seed_number, save_pars = save_pars(all = save_models),
        sample_prior = FALSE, silent = TRUE, file = paste0(mod_output_wd, "/", id_i,
            "_lm_2"))

    models[[paste0(id_i, "_lm_3")]] <- brm(bf(formula_lm_3, family = gaussian()),
        data = df_i, cores = 4, seed = 143019, save_pars = save_pars(all = save_models),
        sample_prior = FALSE, silent = TRUE, file = paste0(mod_output_wd, "/", id_i,
            "_lm_3"))

    # Return the list of models for the current ID
    return(models)
}

load_rds9(): A custom function to load all rds models in a directory and store in a list. Takes a directory with .rds files

load_rds <- function(model_dw) {
    # List all .rds files in the directory
    model_file_list <- list.files(path = model_dw, pattern = "\\.rds$", full.names = TRUE)

    # Strip extensions and get names
    model_names <- tools::file_path_sans_ext(basename(model_file_list))

    # Read all .rds files into a named list
    model_store_list <- setNames(lapply(model_file_list, readRDS), model_names)

    return(model_store_list)
}

incremental_regression_bayes_fits(): A custom function for pulling model fits, loo and r2 using loo() and bayes_R2(), respectively. Takes a list of brm models.

# Define Function to Process the data for each ID
incremental_regression_bayes_fits <- function(models) {
  
  loo_results_list <- list()
  
  # Iterate over the names of the models
  for (mod_name in names(models)) {
    # Extract the model
    mod_i <- models[[mod_name]]
    
    # Compute LOO results
    mod_loo_results_i <- loo::loo(mod_i)
    
    # Extract relevant LOO metrics
    elpd_loo_i <- mod_loo_results_i$elpd_loo
    p_loo_i <- mod_loo_results_i$p_loo
    looic_i <- mod_loo_results_i$looic
    
    # Create a data frame with metrics
    df_i <- data.frame(
      elpd_loo = elpd_loo_i,
      p_loo = p_loo_i,
      looic = looic_i,
      model = mod_name
    )
    
    est_i <- tidy(mod_i, effects = "fixed", conf.int = TRUE) %>% 
      dplyr::select(term, estimate, conf.low, conf.high) %>% 
      tidyr::pivot_wider(
        names_from = term,                # Use `term` as column names
        values_from = c(estimate, conf.low, conf.high),  # Values to pivot
        names_sep = "_"                   # Add a separator to column names
      )
    
    df_i <- cbind(df_i, est_i)
    
    # Store the data frame in the list
    loo_results_list[[mod_name]] <- df_i
  }
  
  # Combind 
  loo_results_combined <- bind_rows(loo_results_list)
  
  # Get R2 
  r2_results <- map_dfr(models, ~ as.data.frame(bayes_R2(.x)), .id = "model") %>%
    tibble::remove_rownames()
  
  # Combind R2 and loo results 
  model_fit_df <- dplyr::full_join(loo_results_combined, r2_results, by = "model") %>% 
    dplyr::select(model, everything()) %>% 
    dplyr::rename(r2 = Estimate,
                  r2_error = Est.Error,
                  r2_q2.5 = Q2.5,
                  r2_q97.5 = Q97.5) %>% 
    dplyr::mutate(id = sub("_(lm_\\d+)$", "", model),
                  model_type = sub("^.*_(lm_\\d+)$", "\\1", model))
  
  return(model_fit_df)
}

bayes_mod_predictions(): This function extracts the predicted values using fitted() from a list of models and combines them with the original data file used for the model. These are the posterior mean fitted values (i.e. the expected value of the response variable given the predictor variables and the estimated posterior distributions of the parameters) for each observation in the dataset, along with 95% credible intervals.

bayes_mod_predictions <- function(models, original_data) {

    prediction_list <- list()

    for (mod_name in names(models)) {
        # Extract mod
        mod_i <- models[[mod_name]]

        # Make mode predictions
        model_predictions_i <- as.data.frame(fitted(mod_i, summary = TRUE)) %>%
            dplyr::mutate(model = mod_name, id = sub("_(lm_\\d+)$", "", mod_name),
                model_type = sub("^.*_(lm_\\d+)$", "\\1", mod_name)) %>%
            dplyr::rename(pred_lower = Q2.5, pred_upper = Q97.5, predicted = Estimate,
                pred_error = Est.Error) %>%
            dplyr::select(model, everything())

        id_i <- model_predictions_i$id[1]

        original_data_i <- original_data %>%
            dplyr::filter(id == id_i) %>%
            dplyr::select(-id)

        model_predictions_original_i <- cbind(model_predictions_i, original_data_i)

        prediction_list[[mod_name]] <- model_predictions_original_i
    }
    predictions_df <- bind_rows(prediction_list)
    return(predictions_df)
}

calcSMR(): authored by Denis Chabot used to estimate SMR with several different methods Claireaux and Chabot (2016) [1]

[1] Claireaux, G. and Chabot, D. (2016) Responses by fishes to environmental hypoxia: integration through Fry’s concept of aerobic metabolic scope. Journal of Fish Biology https://doi.org/10.1111/jfb.12833

calcSMR = function(Y, q = c(0.1, 0.15, 0.2, 0.25, 0.3), G = 1:4) {
    u = sort(Y)
    the.Mclust <- Mclust(Y, G = G)
    cl <- the.Mclust$classification
    # sometimes, the class containing SMR is not called 1 the following
    # presumes that when class 1 contains > 10% of cases, it contains SMR,
    # otherwise we take class 2
    cl2 <- as.data.frame(table(cl))
    cl2$cl <- as.numeric(levels(cl2$cl))
    valid <- cl2$Freq >= 0.1 * length(time)
    the.cl <- min(cl2$cl[valid])
    left.distr <- Y[the.Mclust$classification == the.cl]
    mlnd = the.Mclust$parameters$mean[the.cl]
    CVmlnd = sd(left.distr)/mlnd * 100
    quant = quantile(Y, q)
    low10 = mean(u[1:10])
    low10pc = mean(u[6:(5 + round(0.1 * (length(u) - 5)))])
    # remove 5 outliers, keep lowest 10% of the rest, average Herrmann & Enders
    # 2000
    return(list(mlnd = mlnd, quant = quant, low10 = low10, low10pc = low10pc, cl = cl,
        CVmlnd = CVmlnd))
}

calcO2crit(): authored by Denis Chabot used to estimate O2crit (Pcript). Claireaux and Chabot (2016) [1]

Note: O2 is assumed to be in percentage of dissolved oxygen (DO)

calcO2crit <- function(Data, SMR, lowestMO2 = NA, gapLimit = 4, max.nb.MO2.for.reg = 20) {
    # AUTHOR: Denis Chabot, Institut Maurice-Lamontagne, DFO, Canada first
    # version written in June 2009 last updated in January 2015
    method = "LS_reg"  # will become 'through_origin' if intercept is > 0
    if (is.na(lowestMO2))
        lowestMO2 = quantile(Data$MO2[Data$DO >= 80], p = 0.05)
    # Step 1: identify points where MO2 is proportional to DO
    geqSMR = Data$MO2 >= lowestMO2
    pivotDO = min(Data$DO[geqSMR])
    lethal = Data$DO < pivotDO
    N_under_SMR = sum(lethal)  # points available for regression?
    final_N_under_SMR = lethal  # some points may be removed at Step 4
    lastMO2reg = Data$MO2[Data$DO == pivotDO]  # last MO2 when regulating
    if (N_under_SMR > 1)
        theMod = lm(MO2 ~ DO, data = Data[lethal, ])
    # Step 2, add one or more point at or above SMR 2A, when there are fewer
    # than 3 valid points to calculate a regression
    if (N_under_SMR < 3) {
        missing = 3 - sum(lethal)
        not.lethal = Data$DO[geqSMR]
        DOlimit = max(sort(not.lethal)[1:missing])  # highest DO acceptable
        # to reach a N of 3
        addedPoints = Data$DO <= DOlimit
        lethal = lethal | addedPoints
        theMod = lm(MO2 ~ DO, data = Data[lethal, ])
    }
    # 2B, add pivotDO to the fit when Step 1 yielded 3 or more values?
    if (N_under_SMR >= 3) {
        lethalB = Data$DO <= pivotDO  # has one more value than 'lethal'
        regA = theMod
        regB = lm(MO2 ~ DO, data = Data[lethalB, ])
        large_slope_drop = (coef(regA)[2]/coef(regB)[2]) > 1.1  # arbitrary
        large_DO_gap = (max(Data$DO[lethalB]) - max(Data$DO[lethal])) > gapLimit
        tooSmallMO2 = lastMO2reg < SMR
        if (!large_slope_drop & !large_DO_gap & !tooSmallMO2)
            {
                lethal = lethalB
                theMod = regB
            }  # otherwise we do not accept the additional point
    }
    # Step 3 if the user wants to limit the number of points in the regression
    if (!is.na(max.nb.MO2.for.reg) & sum(lethal) > max.nb.MO2.for.reg) {
        Ranks = rank(Data$DO)
        lethal = Ranks <= max.nb.MO2.for.reg
        theMod = lm(MO2 ~ DO, data = Data[lethal, ])
        final_N_under_SMR = max.nb.MO2.for.reg
    }
    # Step 4
    predMO2 = as.numeric(predict(theMod, data.frame(DO = Data$DO)))
    Data$delta = (Data$MO2 - predMO2)/predMO2 * 100  # residuals set to zero
    # when below pivotDO
    Data$delta[Data$DO < pivotDO | lethal] = 0
    tol = 0  # any positive residual is unacceptable
    HighValues = Data$delta > tol
    Ranks = rank(-1 * Data$delta)
    HighMO2 = HighValues & Ranks == min(Ranks)  # keep largest residual
    if (sum(HighValues) > 0)
        {
            nblethal = sum(lethal)
            Data$W = NA
            Data$W[lethal] = 1/nblethal
            Data$W[HighMO2] = 1
            theMod = lm(MO2 ~ DO, weight = W, data = Data[lethal | HighMO2, ])
            # This new regression is always an improvement, but there can still
            # be points above the line, so we repeat
            predMO2_2 = as.numeric(predict(theMod, data.frame(DO = Data$DO)))
            Data$delta2 = (Data$MO2 - predMO2_2)/predMO2_2 * 100
            Data$delta2[Data$DO < pivotDO] = 0
            tol = Data$delta2[HighMO2]
            HighValues2 = Data$delta2 > tol
            if (sum(HighValues2) > 0)
                {
                  Ranks2 = rank(-1 * Data$delta2)
                  HighMO2_2 = HighValues2 & Ranks2 == 1  # keep the largest residual
                  nblethal = sum(lethal)
                  Data$W = NA
                  Data$W[lethal] = 1/nblethal
                  Data$W[HighMO2_2] = 1
                  theMod2 = lm(MO2 ~ DO, weight = W, data = Data[lethal | HighMO2_2,
                    ])
                  # is new slope steeper than the old one?
                  if (theMod2$coef[2] > theMod$coef[2]) {
                    theMod = theMod2
                    HighMO2 = HighMO2_2
                  }
                }  # end second search for high value
        }  # end first search for high value
    Coef = coefficients(theMod)
    # Step 5, check for positive intercept
    AboveOrigin = Coef[1] > 0
    # if it is, we use a regression that goes through the origin
    if (AboveOrigin) {
        theMod = lm(MO2 ~ DO - 1, data = Data[lethal, ])
        Coef = c(0, coefficients(theMod))  # need to add the intercept (0)
        # manually to have a pair of coefficients
        method = "through_origin"
        HighMO2 = rep(FALSE, nrow(Data))  # did not use the additional value
        # from Step 4
    }
    po2crit = as.numeric(round((SMR - Coef[1])/Coef[2], 1))
    sum_mod = summary(theMod)
    anov_mod = anova(theMod)
    O2CRIT = list(o2crit = po2crit, SMR = SMR, Nb_MO2_conforming = N_under_SMR, Nb_MO2_conf_used = final_N_under_SMR,
        High_MO2_required = sum(HighMO2) == 1, origData = Data, Method = method,
        mod = theMod, r2 = sum_mod$r.squared, P = anov_mod$"Pr(>F)", lethalPoints = which(lethal),
        AddedPoints = which(HighMO2))
}  # end function

plotO2crit(): authored by Denis Chabot, used to plot the modes used for the calcO2crit() function. Claireaux and Chabot (2016) [1]

Note: I added abline(h=lowestMO2, col=“pink”) so that I could visualise the lowestMO2 position

plotO2crit <- function(o2critobj, plotID = "", Xlab = "Dissolved oxygen (% sat.)",
    Ylab = "dotitalumol", smr.cex = 0.9, o2crit.cex = 0.9, plotID.cex = 1.2, Transparency = T,
    ...) {
    # AUTHOR: Denis Chabot, Institut Maurice-Lamontagne, DFO, Canada first
    # version written in June 2009 last updated 2015-02-09 for R plotting
    # devices that do not support transparency (e.g., postscript), set
    # Transparency to FALSE
    smr = o2critobj$SMR
    if (Ylab %in% c("dotitalumol", "italumol", "dotumol", "umol", "dotitalmg", "italmg",
        "dotmg", "mg")) {
        switch(Ylab, dotitalumol = {
            mo2.lab = expression(paste(italic(dot(M))[O[2]], " (", mu, "mol ", O[2],
                " ", min^-1, " ", kg^-1, ")"))
        }, italumol = {
            mo2.lab = expression(paste(italic(M)[O[2]], " (", mu, "mol ", O[2], " ",
                min^-1, " ", kg^-1, ")"))
        }, dotumol = {
            mo2.lab = expression(paste(dot(M)[O[2]], " (", mu, "mol ", O[2], " ",
                min^-1, " ", kg^-1, ")"))
        }, umol = {
            mo2.lab = expression(paste(M[O[2]], " (", mu, "mol ", O[2], " ", min^-1,
                " ", kg^-1, ")"))
        }, dotitalmg = {
            mo2.lab = expression(paste(italic(dot(M))[O[2]], " (mg ", O[2], " ",
                h^-1, " ", kg^-1, ")"))
        }, italmg = {
            mo2.lab = expression(paste(italic(M)[O[2]], " (mg ", O[2], " ", h^-1,
                " ", kg^-1, ")"))
        }, dotmg = {
            mo2.lab = expression(paste(dot(M)[O[2]], " (mg ", O[2], " ", h^-1, " ",
                kg^-1, ")"))
        }, mg = {
            mo2.lab = expression(paste(M[O[2]], " (mg ", O[2], " ", h^-1, " ", kg^-1,
                ")"))
        })
    } else mo2.lab = Ylab
    if (Transparency) {
        Col = c(rgb(0, 0, 0, 0.7), "red", "orange")
    } else {
        Col = c(grey(0.3), "red", "orange")
    }
    Data = o2critobj$origData
    lowestMO2 = quantile(Data$MO2[Data$DO >= 80], p = 0.05)  # I added this
    Data$Color = Col[1]
    Data$Color[o2critobj$lethalPoints] = Col[2]
    Data$Color[o2critobj$AddedPoints] = Col[3]
    # ordinary LS regression without added points: blue line, red symbols
    # ordinary LS regression with added points: blue line, red & orange symbols
    # regression through origin: green dotted line, red symbols
    line.color = ifelse(o2critobj$Method == "LS_reg", "blue", "darkgreen")
    line.type = ifelse(o2critobj$Method == "LS_reg", 1, 3)
    limX = c(0, max(Data$DO))
    limY = c(0, max(Data$MO2))
    plot(MO2 ~ DO, data = Data, xlim = limX, ylim = limY, col = Data$Color, xlab = Xlab,
        ylab = mo2.lab, ...)
    coord <- par("usr")
    if (plotID != "") {
        text(0, coord[4], plotID, cex = plotID.cex, adj = c(0, 1.2))
    }
    abline(h = lowestMO2, col = "pink")  # I added this
    abline(h = smr, col = "orange")
    text(coord[1], smr, "SMR", adj = c(-0.1, 1.3), cex = smr.cex)
    text(coord[1], smr, round(smr, 1), adj = c(-0.1, -0.3), cex = smr.cex)
    if (!is.na(o2critobj$o2crit)) {
        abline(o2critobj$mod, col = line.color, lty = line.type)
        segments(o2critobj$o2crit, smr, o2critobj$o2crit, coord[3], col = line.color,
            lwd = 1)
        text(x = o2critobj$o2crit, y = 0, o2critobj$o2crit, col = line.color, cex = o2crit.cex,
            adj = c(-0.1, 0.5))
    }
}  # end of function

6 📂 Directories

6.1 Input

📥 input_data_wd: Directory for the metadata

wd <- getwd()
input_data_wd <- paste0(wd, "./input-data")  # creates a variable with the name of the wd we want to use

📥 mod_data_wd: Directory for model output data estimated slopes

mod_data_wd <- paste0(wd, "./mod-data")

6.2 Output

📤 output_fig_wd: this is where we will put the figures

output_fig_wd <- paste0(wd, "./output-fig")
if (!dir.exists("output-fig")) {
    dir.create("output-fig")
}

📤 output_mods_wd: this is where we will put the models

output_mods_wd <- paste0(wd, "./output-mod")
if (!dir.exists("output-mod")) {
    dir.create("output-mod")
}

7 💿 Data

7.1 Slopes (ṀO)

💿 slope_df: We have imported the slopes extracted in LabChart during each phase of the experiment

 setwd(input_data_wd)

data_file <- "labchart-slope-data.xlsx"

# 
# # Get the names of all sheets in the Excel file
sheet_names <- readxl::excel_sheets(data_file)
all_trials_select <- c("start_date", "order", "phase", "cycle", "date", "time")
slope_list <- list()

for (sheet in sheet_names) {

  df <- read_excel(data_file, sheet = sheet) %>% 
  dplyr::rename_with(tolower)
  
a_name <- paste0("a_", tolower(sheet))
a_df <- df %>%
  dplyr::select(starts_with('a'), all_trials_select) %>% 
  dplyr::rename(temp = a_temp) %>% 
  dplyr::mutate(across(starts_with('a'), as.numeric)) %>% 
  pivot_longer(
    cols = starts_with('a'), # Select all columns to pivot
    names_to = c("chamber_id", ".value"), # Separate column names into 'id' and other variables
    names_sep = "_"
  ) %>%
  dplyr::mutate(respirometer_group = "a") # Add a new column with a fixed value

slope_list[[a_name]]<- a_df

b_name <- paste0("b_", tolower(sheet))
b_df <- df %>% 
  dplyr::select(starts_with('b'), all_trials_select) %>% 
  dplyr::rename(temp = b_temp) %>% 
  dplyr::mutate(across(starts_with('b'), as.numeric)) %>% 
  pivot_longer(
    cols = starts_with('b'), # Select all columns to pivot
    names_to = c("chamber_id", ".value"), # Separate column names into 'id' and other variables
    names_sep = "_"
  ) %>% 
    dplyr::mutate(respirometer_group = "b")

slope_list[[b_name]] <- b_df

c_name <- paste0("c_", tolower(sheet))
c_df <- df %>% 
  dplyr::select(starts_with('c'), all_trials_select) %>% 
  dplyr::rename(temp = c_temp,
                i_cycle = cycle) %>% 
  dplyr::mutate(across(starts_with('c'), as.numeric)) %>%
  pivot_longer(
    cols = starts_with('c'), # Select all columns to pivot
    names_to = c("chamber_id", ".value"), # Separate column names into 'id' and other variables
    names_sep = "_"
  ) %>% 
    dplyr::mutate(respirometer_group = "c") %>% 
  dplyr::rename(cycle = i_cycle)

slope_list[[c_name]] <- c_df

d_name <- paste0("d_", tolower(sheet))
d_df <- df %>% 
  dplyr::select(starts_with('d'), all_trials_select) %>% 
  dplyr::rename(temp = d_temp,
                i_date = date) %>% 
  dplyr::mutate(across(starts_with('d'), as.numeric)) %>%
  pivot_longer(
    cols = starts_with('d'), # Select all columns to pivot
    names_to = c("chamber_id", ".value"), # Separate column names into 'id' and other variables
    names_sep = "_"
  ) %>% 
    dplyr::mutate(respirometer_group = "d") %>% 
  dplyr::rename(date = i_date)

slope_list[[d_name]] <- d_df
}


slope_df <- bind_rows(slope_list) %>% 
  dplyr::mutate(resp_cat_date = paste0(respirometer_group, "_", start_date),
                chamber_n = str_extract(chamber_id, "\\d+"),
                id_prox = paste0(resp_cat_date, "_", chamber_n),
                time_hms = as_hms(time*3600),
                date_chr = format(date, "%d/%m/%Y")
                )

7.2 Fish metadata

💿 metadata: This is the meta data for each chamber

Note: We are also adding volume based on chamber type.

setwd(input_data_wd)

metadata <- read_excel("fish-meta.xlsx", na = "NA") %>%
    dplyr::mutate(id_split = id) %>%
    tidyr::separate(id_split, into = c("respirometer_group", "salinity_group", "start_date",
        "chamber"), sep = "_") %>%
    dplyr::mutate(volume = dplyr::case_when(chamber_type == "L" ~ 0.3, chamber_type ==
        "M_M" ~ 0.105, chamber_type == "M_NM" ~ 0.11, chamber_type == "S" ~ 0.058,
        chamber_type == "SM" ~ 0.075, chamber_type == "D3" ~ 0.055, TRUE ~ NA), id_prox = paste0(respirometer_group,
        "_", start_date, "_", chamber), rel_size = mass/volume)

7.3 Urbina, Glover, and Forster (2012)

💿 urbina_et_al_2012: This is the mean level data extracted from Urbina, Glover, and Forster (2012)[2] Figure 1a. We used the metaDigitise package in R to extract the data [3].

setwd(input_data_wd)

urbina_et_al_2012 <- read_excel("urbina-et-al-2012-fig1a.xlsx")

7.4 O₂crit viusal inspection

💿 o2crit_check: This data frame includes the visual based assessment for O₂crit for all fish included in the analysis (n = 58). The inspections were made using figures genbrated by the plotO2crit() function, and can be viewed in combined_chabot_plots.pdf. The visual assessment was done independently by all authors, and the presence of an O₂crits allocated to ‘yes’, ‘no’, ‘maybe’. If yes, or maybe, an estimate of the dissolved oxygen percentage was given.

Directory to the figures

📁 gmac-mo2-24/ └── 📂 output-fig/
└── 📂 model_chabot/ └── 📄 combined_chabot_plots.pdf

setwd(input_data_wd)

o2crit_check <- read_excel("o2crit-visual-assessment.xlsx", na = "NA")

7.4.1 Combinding metadata

Adding the meta data to the slopes data frame

slope_df_2 <- slope_df %>%
    dplyr::select(-start_date, -respirometer_group) %>%
    left_join(metadata, by = "id_prox") %>%
    dplyr::mutate(light_dark = if_else(time_hms >= as.hms("07:00:00") & time_hms <
        as.hms("19:00:00"), "light", "dark")) %>%
    dplyr::arrange(id)

8 🧹 Data tidy

8.1 Numbers

We have 64 fish with MO₂ data

n <- slope_df_2 %>%
    dplyr::filter(chamber_condition == "fish") %>%
    dplyr::distinct(id) %>%
    nrow(.)

paste0("n = ", n)
## [1] "n = 64"

With 48 from the 0 ppt and 48 from 9 ppt groups

slope_df_2 %>%
    dplyr::filter(chamber_condition == "fish") %>%
    dplyr::group_by(salinity_group) %>%
    dplyr::reframe(`n total` = length(unique(id))) %>%
    gt() %>%
    cols_label(salinity_group = "Salinity group") %>%
    cols_align(align = "center", columns = everything())
Salinity group n total
0 33
9 31

8.2 Fish size

Here we calculate the mean length and size of fish used in the experiment.

mass_length <- slope_df_2 %>%
    dplyr::filter(chamber_condition == "fish") %>%
    dplyr::group_by(id) %>%
    dplyr::sample_n(1) %>%
    dplyr::ungroup() %>%
    dplyr::reframe(x_mass = round(mean(mass, na.rm = TRUE), 3), min_mass = round(min(mass,
        na.rm = TRUE), 3), max_mass = round(max(mass, na.rm = TRUE), 3), x_length = round(mean(length,
        na.rm = TRUE), 2), min_length = round(min(length, na.rm = TRUE), 2), max_length = round(max(length,
        na.rm = TRUE), 2))

mass_mean <- mass_length %>%
    pull(x_mass)

mass_min <- mass_length %>%
    pull(min_mass)

mass_max <- mass_length %>%
    pull(max_mass)

length_mean <- mass_length %>%
    pull(x_length)

length_min <- mass_length %>%
    pull(min_length)

length_max <- mass_length %>%
    pull(max_length)

paste0("The mean mass of fish was ", mass_mean, " g (range: ", mass_min, "-", mass_max,
    ")", ", and the mean length was ", length_mean, " mm (range: ", length_min, "-",
    length_max, ")")
## [1] "The mean mass of fish was 0.532 g (range: 0.21-1.6), and the mean length was 50.41 mm (range: 40-70)"

8.3 Filtering trials

We will remove 6 trials which had errors. These are as follows:

  • a_0_25nov_3 fish died during trial
  • b_0_26nov_4 flat lined early
  • c_0_22nov_2 chamber was opened early
  • c_9_26nov_2 chamber was opened early
  • c_9_26nov_4 chamber was opened early
  • d_9_27nov_3 sensor was jumpy and end points were hard to confidently ID visually
remove_trial_error <- c("a_0_25nov_3", "b_0_26nov_4", "c_0_22nov_2", "c_9_26nov_2",
    "c_9_26nov_4", "d_9_27nov_3")

slope_df_filter <- slope_df_2 %>%
    dplyr::filter(!(id %in% remove_trial_error))

We now have 58 fish with MO2 data

n <- slope_df_filter %>%
    dplyr::filter(chamber_condition == "fish") %>%
    dplyr::distinct(id) %>%
    nrow(.)

paste0("n = ", n)
## [1] "n = 58"

With 30 in the 0 ppt group and 28 in the 9 ppt group

slope_df_filter %>%
    dplyr::filter(chamber_condition == "fish") %>%
    dplyr::group_by(salinity_group) %>%
    dplyr::reframe(`n total` = length(unique(id))) %>%
    gt() %>%
    cols_label(salinity_group = "Salinity group") %>%
    cols_align(align = "center", columns = everything())
Salinity group n total
0 30
9 28

Chamber size breakdown

slope_df_filter %>%
    dplyr::filter(chamber_condition == "fish") %>%
    dplyr::mutate(volume = if_else(volume == 0.055, 0.058, volume)) %>%
    dplyr::group_by(volume) %>%
    dplyr::reframe(`n total` = length(unique(id)), median_mass = median(mass, na.rm = TRUE),
        sd_mass = sd(mass, na.rm = TRUE)) %>%
    gt() %>%
    cols_label(volume = "Chamber type", median_mass = "Median mass", sd_mass = "Standard deviation mass") %>%
    cols_align(align = "center", columns = everything())
Chamber type n total Median mass Standard deviation mass
0.058 36 0.43 0.1496554
0.105 21 0.61 0.2561871
0.300 1 1.60 0.0000000

Numbers per trial approach

phases <- c("50c", "100c")

slope_df_filter %>%
    dplyr::filter(chamber_condition == "fish" & phase %in% phases) %>%
    dplyr::group_by(phase) %>%
    dplyr::reframe(`n total` = length(unique(id))) %>%
    gt() %>%
    cols_label(phase = "Type of trial", ) %>%
    cols_align(align = "center", columns = everything())
Type of trial n total
100c 19
50c 39

8.4 Filtering MO₂ estimates

Here we apply the following filters to the MO₂ data:

  • Remove the first 5 SMR cycles (burn in)
  • Remove all positive raw slopes
  • Remove all MO₂ calculated using less then 60 data points (5 min at 0.2 Hz)
  • Remove all MO₂ calculated if O₂ increases in a closed phase (i.e. trial has ended)

Check positive values for MO₂ before removing.

slope_tidy_remove_flush <- slope_df_filter %>%
  dplyr::filter(phase != "smr", chamber_condition == "fish") %>% 
  dplyr::group_by(id) %>%
  dplyr::arrange(id, order) %>%  # Ensure the data is sorted within each group
  dplyr::mutate(o2_diff = if_else(row_number() == 1, 0, o2 - lag(o2)), # Calculate the difference in 'o2'
                o2_diff_cumsum = cumsum(o2_diff > 1)) %>%  # Checks first occurrence and sums 
  dplyr::filter(o2_diff_cumsum == 0) %>%  # Keep rows until the first jump > 1
  dplyr::ungroup() %>%
  dplyr::select(-o2_diff, -o2_diff_cumsum)

postive_slopes <- slope_tidy_remove_flush %>%
  dplyr::filter(chamber_condition == "fish" & mo2corr > 0)

n_postive_slopes <- nrow(postive_slopes)

n_slopes <-  nrow(slope_tidy_remove_flush)

list_postive_all <- postive_slopes %>% 
  dplyr::distinct(id) %>% 
  dplyr::pull(id)

print(paste0("There are ", length(list_postive_all), " fish with postive slopes. These fish are: ", paste(list_postive_all, collapse = ", "), ". For all estimated slopes (n = ", n_slopes, ") ", round((n_postive_slopes/n_slopes)*100,2) , "% were postive (n = ", n_postive_slopes, ")"))
## [1] "There are 13 fish with postive slopes. These fish are: a_0_24nov_1, b_0_24nov_3, b_9_21nov_2, b_9_21nov_3, b_9_22nov_1, b_9_22nov_2, b_9_22nov_3, c_0_21nov_1, c_0_21nov_2, c_0_22nov_4, c_9_24nov_3, d_0_22nov_2, d_9_25nov_3. For all estimated slopes (n = 1105) 1.27% were postive (n = 14)"

Filtering the MO₂ data

cycle_burn <- 0:4

slope_df_filter_1 <- slope_df_filter %>%
  dplyr::filter(!(cycle %in% cycle_burn) & 
                  mo2corr < 0 & 
                  n > 60 &
                  chamber_condition == "fish"
                )
  
# Filter out the end flush
slope_tidy_closed <- slope_df_filter_1 %>%
  dplyr::filter(phase != "smr") %>% 
  dplyr::group_by(id) %>%
  dplyr::arrange(id, order) %>%  # Ensure the data is sorted within each group
  dplyr::mutate(o2_diff = if_else(row_number() == 1, 0, o2 - lag(o2)), # Calculate the difference in 'o2'
                o2_diff_cumsum = cumsum(o2_diff > 1)) %>%  # Checks first occurrence and sums 
  dplyr::filter(o2_diff_cumsum == 0) %>%  # Keep rows until the first jump > 1
  dplyr::ungroup() %>%
  dplyr::select(-o2_diff, -o2_diff_cumsum)

slope_tidy_smr <- slope_df_filter_1 %>% 
  dplyr::filter(phase == "smr")

slope_df_filter_2 <- rbind(slope_tidy_smr, slope_tidy_closed) %>% 
  dplyr::arrange(id, order)

8.5 Calculating SMR

We will calculate SMR using calcSMR function by Chabot, Steffensen and Farrell (2016)[1]. Specifically, we use mean of the lowest normal distribution (MLND) where CVmlnd < 5.4 and the mean of the lower 20% quantile (q0.2) were CVmlnd > 5.4. If CVmlnd is not calculated we have used q0.2.

labchart_chabot_smr <- slope_df_filter_2 %>%
    dplyr::filter(phase == "smr")

# Extract distinct IDs
ids <- labchart_chabot_smr %>%
    dplyr::distinct(id) %>%
    dplyr::pull()

# Initialise an empty list to store SMR data
smr_list <- list()

# Process each ID
for (id_i in ids) {
    tryCatch({
        # Filter the data for the current ID
        df_i <- labchart_chabot_smr %>%
            dplyr::filter(id == id_i) %>%
            dplyr::mutate(abs_mo2corr = abs(mo2corr))

        # Calculate SMR results
        calcSMR_results <- calcSMR(df_i$abs_mo2corr)
        CVmlnd_i <- calcSMR_results$CVmlnd
        quant_i <- calcSMR_results$quant %>%
            as_tibble()
        quant_20per_i <- quant_i$value[3]
        mlnd_i <- calcSMR_results$mlnd
        smr_value <- if_else(CVmlnd_i < 5.4, mlnd_i, quant_20per_i)
        smr_type <- if_else(CVmlnd_i < 5.4, "mlnd", "quant_20per")
        smr_value <- if_else(is.na(smr_value), quant_20per_i, smr_value)
        smr_type <- if_else(is.na(smr_type), "quant_20per", smr_type)

        # Create a data frame for the current ID
        smr_df_i <- tibble::tibble(id = id_i, smr = smr_value, smr_est = smr_type)

    }, error = function(e) {
        # Handle errors by assigning NA values
        smr_df_i <- tibble::tibble(id = id_i, smr = NA, smr_est = NA)
    })

    # Append to the list
    smr_list[[id_i]] <- smr_df_i
}

# Combine all individual SMR data frames into one
smr_df <- bind_rows(smr_list) %>%
    dplyr::rename(smr_chabot = smr, smr_chabot_method = smr_est)

slope_df_filter_3 <- slope_df_filter_2 %>%
    dplyr::left_join(., smr_df, by = "id")

8.6 Transforming MO₂

Here we are transforming the MO₂ units. The resulting values are as follows:

  • MO2 is absolute value of the background and leak corrected MO₂ slope from Labchart (mo2corr) times the net volume of the chamber (volume - fish mass), × 60, × 60, to achieve MO₂ as mg-1 O₂ h-1

  • MO2_g is MO2 divided by fish mass to achieve MO₂ as mg-1 O₂ g-1 h-1 (i.e. mass standardised)

  • SMR absolute value of the SMR estimates using methods described by Chabot, Steffensen and Farrell (2016)[1] times the net volume of the chamber (volume - fish mass), × 60, × 60, to achieve SMR as mg-1 O₂ h-1)

  • SMR_g is SMR divided by fish mass to achieve SMR as mg-1 O₂ g-1 h-1 (i.e. mass standardised)

  • DO is dissolved oxygen percentage calculated from O₂ values (mg-1 L-1) using the recorded temperature, salinity, and a constant atmospheric pressure (Pa; 1013.25)

  • o2_kpa is the O₂ concentration in kilopascal (kpa). This is used to make a comparative figure only.

# Combine back into one data frame
slope_tidy <- slope_df_filter_3 %>% 
    dplyr::mutate(DO = conv_o2(
                   o2 = o2,
                   from = "mg_per_l",
                   to = "percent_a.s.",
                   temp = temp, #C
                   sal = measured_salinity,
                   atm_pres = 1013.25),
                  o2_kpa = conv_o2(
                   o2 = o2,
                   from = "mg_per_l",
                   to = "kPa",
                   temp = temp, #C
                   sal = measured_salinity,
                   atm_pres = 1013.25),
                  net_volume = volume - mass/1000,
                  MO2 = abs(mo2corr)*net_volume*60*60,
                  MO2_g = MO2/mass,
                  SMR = abs(smr_chabot)*net_volume*60*60,
                  SMR_g = SMR/mass
                  )

Trial lengths

slope_tidy %>%
    dplyr::filter(phase != "smr") %>%
    dplyr::mutate(datetime = as.POSIXct(paste(date, time_hms), format = "%Y-%m-%d %H:%M:%S")) %>%
    dplyr::group_by(id) %>%
    dplyr::reframe(phase = phase[1], start_time = min(datetime, na.rm = TRUE), end_time = max(datetime,
        na.rm = TRUE), time_diff = difftime(end_time, start_time, units = "hours")  # Change units as needed
) %>%
    dplyr::group_by(phase) %>%
    dplyr::reframe(`Minimum duration` = round(min(time_diff), 2), `Maximum duration` = round(max(time_diff),
        2), `Median duration` = round(median(time_diff), 2)) %>%
    dplyr::mutate(phase = if_else(phase == "100c", "Closed at 100% air saturation",
        "Closed at 75% and 50% air saturation")) %>%
    dplyr::rename(Approach = phase) %>%
    gt()
Approach Minimum duration Maximum duration Median duration
Closed at 100% air saturation 3 13 7
Closed at 75% and 50% air saturation 2.91 8.02 5.96

9 📊 Exploratory Data Analysis (EDA)

9.1 O₂ vs MO₂

9.1.1 Figure S1

This interactive was used to identify any outliers, or potential errors.

lm_lines <- slope_tidy %>%
  group_by(id) %>%
  summarise(
    DO_seq = list(seq(min(DO), max(DO), length.out = 100)),  # Generate a sequence of DO values
    MO2_pred = list(predict(lm(MO2_g ~ DO, data = cur_data()), newdata = data.frame(DO = seq(min(DO), max(DO), length.out = 100))))
  ) %>%
  unnest(c(DO_seq, MO2_pred))  # Expand lists into rows

# Create scatter plot with markers for each fish
p <- plot_ly(
  data = slope_tidy,
  x = ~DO,
  y = ~MO2_g,
  type = "scatter",
  mode = "markers",
  color = ~id,  # Colour points by fish ID
  marker = list(opacity = 0.6),
  name = ~id
)

# Add regression lines for each fish
p <- p %>%
  add_trace(
    data = lm_lines,
    x = ~DO_seq,
    y = ~MO2_pred,
    type = "scatter",
    mode = "lines",
    color = ~id,  # Ensure each line matches its corresponding fish
    line = list(width = 1, dash = "solid"),
    showlegend = FALSE  # Avoid cluttering the legend
  )

# Final layout
p <- p %>%
  layout(
    title = "MO<sub>2</sub> vs Dissolved Oxygen with individual linear regressions",
    xaxis = list(title = "Dissolved Oxygen (%)"),
    yaxis = list(title = "MO<sub>2</sub> (mg<sup>-1</sup> O<sub>2</sub> g<sup>-1</sup> h<sup>-1</sup>)"),
    showlegend = FALSE
  )

# Display plot
p

Figure S1: interactive plot of metabolic rate measurements (MO₂; mg O₂ g-1h-1) by dissolved oxygen percentage (DO) for all fish, including all estimates during the SMR phase (i.e. intermittent phase). Individual linear regression were fitted for visual reference, and do not represent the best fitting regression.

9.1.2 Figure S2

Looking at the difference responses in the two salinity groups.

salinity_summary <- slope_tidy %>% 
  dplyr::group_by(salinity_group) %>% 
  dplyr::reframe(n = length(unique(id)))

slope_tidy %>% 
  ggplot(aes(y = MO2_g, x = DO, colour = id)) + # Default aesthetics
  geom_point(show.legend = FALSE) +
  geom_smooth(aes(group = id), method = "lm", se = FALSE, colour = scales::alpha("black", 0.5)) + # Transparent black lines
  geom_smooth(method = "lm", se = TRUE, colour = "red") + # Overall smooth line
  geom_smooth(method = "gam", formula = y ~ s(x, bs = "cs"), se = TRUE, colour = "red", linetype = "dashed") +
  theme_clean() +
  facet_wrap(~salinity_group) +
  labs(
    subtitle = "mo2 vs o2 by salinity treatment",
    x = "Dissolved oxygen percentage (DO)",
    y = "MO2 (O2 mg/g/h)"
  ) +
  geom_text(data = salinity_summary,
            aes(x = -Inf, y = Inf, label = paste0("italic(n) == ", n)),
            hjust = -0.1, vjust = 1.2, inherit.aes = FALSE, parse = TRUE)

Figure S2: Metabolic rate measurements (MO₂; mg O₂ g-1h-1) by dissolved oxygen percentage (DO) for fish from the two salinity treatments, including all estimates during the SMR phase (i.e. intermittent phase). Individual linear regression were fitted for visual reference, and do not represent the best fitting regression. The solid red and dashed red line represents the global trend in each of the treatments, both a linear (solid red) and non-linear (dashed red; Generalized Additive Model fitted with geom_smooth).

9.1.3 Figure S3

A plot to look at the different chamber types

chamber_summary <- slope_tidy %>% 
  dplyr::group_by(chamber_type) %>% 
  dplyr::reframe(n = length(unique(id)))

slope_tidy %>% 
  ggplot(aes(y = MO2_g, x = DO, colour = id)) + # Default aesthetics
  geom_point(show.legend = FALSE) +
  geom_smooth(aes(group = id), method = "lm", se = FALSE, colour = scales::alpha("black", 0.5)) + # Transparent black lines
  geom_smooth(method = "lm", se = TRUE, colour = "red") + # Overall smooth line
  geom_smooth(se = TRUE, colour = "red", linetype = "dashed") +
  theme_clean() +
  facet_wrap(~chamber_type, scale = "free") +
  labs(
    subtitle = "mo2 vs o2 by chamber type",
    x = "Dissolved oxygen percentage (DO)",
    y = "MO2 (mg O2 g/h)"
  ) +
  geom_text(data = chamber_summary,
            aes(x = -Inf, y = Inf, label = paste0("italic(n) == ", n)),
            hjust = -0.1, vjust = 1.2, inherit.aes = FALSE, parse = TRUE)

Figure S3: Metabolic rate measurements (MO₂; mg O₂ g-1h-1) by dissolved oxygen percentage (DO) for fish tested in the 4 different chamber types, including all estimates during the SMR phase (i.e. intermittent phase). Individual linear regression were fitted for visual reference, and do not represent the best fitting regression. The solid red and dashed red line represents the global trend in each of the treatments, both a linear (solid red) and non-linear (dashed red; Generalized Additive Model fitted with geom_smooth).

9.2 Routine MO₂

Making an SMR phase only data frame

slope_tidy_smr <- slope_tidy %>%
    dplyr::filter(phase == "smr")

9.2.1 Figure S4

Routine MO2 by salinity

mean_mo2_salinity <- slope_tidy_smr %>% 
  dplyr::group_by(salinity_group) %>% 
  dplyr::reframe(mean_mo2 = mean(MO2, na.rm = TRUE))

fig_i <- ggplot() +
    geom_violin(data = slope_tidy_smr, aes(x = salinity_group, y = MO2, fill = salinity_group), color = NA, alpha = 0.3) +
  geom_jitter(data = slope_tidy_smr, aes(x = salinity_group, y = MO2, fill = salinity_group),
                       shape = 21, size = 2, color = "black", alpha = 0.2) +
  geom_boxplot(data = slope_tidy_smr, aes(x = salinity_group, y = MO2, fill = salinity_group),
                        size = 1, alpha = 0.5, outlier.shape = NA, width = 0.3) +
  geom_point(data = mean_mo2_salinity, 
                aes(x = salinity_group, y = mean_mo2, fill = salinity_group), 
                size = 3, alpha = 0.8, colour = "black", stroke = 2) +
  scale_fill_manual(values = c("#4B5320", "#000080")) +  # Custom fill colours
  scale_colour_manual(values = c("#4B5320", "#000080")) +
  theme_clean() +
  theme(legend.position = "none") +
  labs(
    subtitle = "",
    x = "Salinity group (ppt)",
    y = "Routine MO2 (mg O2 g/h)"
  )

fig_i

Figure S4: Plot of all MO2 measures during SMR phase by salinity treatment. The small points are the raw observed values, the shaded area behind the points is the a kernel density of the observed data, the box plot shows the median and interquartile range (IQR), and the large point shows the mean.

9.3 SMR

9.3.1 Figure S5

Here’s the same plot but for only the SMR, as estimated with calcSMR function by Chabot, Steffensen and Farrell (2016)[1]. Specifically, we use mean of the lowest normal distribution (MLND) where CVmlnd < 5.4 and the mean of the lower 20% quantile (q0.2) were CVmlnd > 5.4. If CVmlnd is not calculated we have used q0.2.

smr_only <- slope_tidy_smr %>% 
  dplyr::group_by(id) %>% 
  dplyr::slice(1)

mean_smr_salinity <- smr_only %>% 
  dplyr::group_by(salinity_group) %>% 
  dplyr::reframe(mean_smr = mean(SMR, na.rm = TRUE))

fig_i <- ggplot() +
    geom_violin(data = smr_only, aes(x = salinity_group, y = SMR, fill = salinity_group), color = NA, alpha = 0.3) +
  geom_jitter(data = smr_only, aes(x = salinity_group, y = SMR, fill = salinity_group),
                       shape = 21, size = 2, color = "black", alpha = 0.2) +
  geom_boxplot(data = smr_only, aes(x = salinity_group, y = SMR, fill = salinity_group),
                        size = 1, alpha = 0.5, outlier.shape = NA, width = 0.3) +
  geom_point(data = mean_smr_salinity, 
                aes(x = salinity_group, y = mean_smr, fill = salinity_group), 
                size = 3, alpha = 0.8, colour = "black", stroke = 2) +
  scale_fill_manual(values = c("#4B5320", "#000080")) +  # Custom fill colours
  scale_colour_manual(values = c("#4B5320", "#000080")) +
  theme_clean() +
  theme(legend.position = "none") +
  labs(
    subtitle = "",
    x = "Salinity group (ppt)",
    y = "MO2 (mg O2 g/h)"
  )

fig_i

Figure S5: The standard metabolic rate (SMR) by salinity treatment. The small points are the raw observed values, the shaded area behind the points is the a kernel density of the observed data, the box plot shows the median and interquartile range (IQR), and the large point shows the mean.

9.4 Individual O₂, MO₂, and SMR

Here we will plot the individual relationship between O₂, MO₂, and SMR for all fish

Create output directory if needed

Loop through each fish ID to create a plot, save these to a single PDF file, this is called combined_slopes.pdf

# # Define consistent colors for all phases phase_colors <- c( '50c' =
# '#e56b6f', '75c' = '#b56576', '100c' = '#6d597a', 'smr' = '#355070' ) ids <-
# slope_tidy %>% dplyr::distinct(id) %>% pull(id) %>% as.list() MO2_plot_list
# <- list() # 1) Open the PDF device once pdf( file =
# file.path(output_fig_slopes_wd, 'combined_slopes.pdf'), width = 8, height = 6
# ) # 2) Loop over IDs and create each plot for (id_i in ids) { smr <-
# slope_tidy %>% dplyr::filter(id == id_i) %>% dplyr::slice(1) %>%
# dplyr::pull(SMR) plot <- slope_tidy %>% dplyr::filter(id == id_i) %>%
# ggplot(aes(x = o2, y = MO2)) + geom_hline(yintercept = smr, linetype =
# 'dashed', color = 'darkred') + geom_point(aes(colour = phase)) +
# scale_color_manual(values = phase_colors, drop = FALSE) + # Ensures
# consistent colours theme_clean() + labs( subtitle = paste0(id_i, ' slopes'),
# x = 'Mean o2 (mg_per_l)', y = 'abs(mo2) (mg_per_l)' ) # Instead of saving
# each plot separately, just print it print(plot) MO2_plot_list[[id_i]] <- plot
# } # 3) Close the PDF device *after* the loop dev.off()

Figure S7: Metabolic rate measurements (MO₂; mg O₂ g-1h-1 by O2) at each of the four experimental phase (the over night SMR intermittent-flow respirometry phase, closed phases at 50%, 75% or 100% O2). The estimated SMR value is represented as a dashed red line.

10 🧮 Analysis

10.1 Routine MO2

10.1.1 Scaling predictors

Here we scale our predictors for the model. We have first applied an log transformation to mass, so its directly interpretable as the mass-scaling exponent, just like in metabolic theory.

scale_list <- c("temp", "order", "log_mass")

slope_tidy_smr <- slope_tidy_smr %>%
    dplyr::mutate(log_mass = log(mass), across(all_of(scale_list), ~scale(.x, center = TRUE,
        scale = FALSE), .names = "{.col}_c"), light_dark_c = if_else(light_dark ==
        "light", 0.5, -0.5))

Summary of predictors used in the model below

slope_tidy_smr %>%
    dplyr::mutate(cycle = as.numeric(cycle)) %>%
    dplyr::reframe(temp_range = paste0(round(min(temp, na.rm = TRUE), 2), "–",
        round(max(temp, na.rm = TRUE), 2)), mass_range = paste0(round(min(mass, na.rm = TRUE),
        2), "–", round(max(mass, na.rm = TRUE), 2)), cycle_range = paste0(round(min(cycle,
        na.rm = TRUE), 2), "–", round(max(cycle, na.rm = TRUE), 2)), number_fish = length(unique(id))) %>%
    gt()
temp_range mass_range cycle_range number_fish
13.84–14.38 0.21–1.6 5–27 58

10.1.2 Model structure

Here we will use a Bayesian Generalised Linear Mixed Model (GLMM) with a Gamma distribution and a log link, where the shape parameter (α) is also modelled as a function of predictors. This models MO2 by salinity during the SMR phase to see if the fish held at different salinities have different RMRs. We have also added a few scaled predictors, that may help describe variation in the data, such as mass (g; 0.21–1.6) temperature (°C; 13.84–14.38), cycle order (5–27), and light/dark cycle (light or dark; light between 07:00:00 and 19:00:00), we also include a random effect for fish id to account for multiple MO2 measures on each fish (1–58). We allowed the the shape parameter (α) to vary as a function of some of the predictors (e.g. salinity_group, order_z) to improve fit.

mo2_gamma_bf <- bf(MO2 ~ temp_c + order_c + light_dark_c + log_mass_c + salinity_group +
    (1 | id), shape ~ salinity_group + order_c, family = Gamma(link = "log"))

10.1.3 Prior selection

These are the default priors. We will use these.

default_prior <- get_prior(mo2_gamma_bf, data = slope_tidy_smr, family = Gamma(link = "log"))
default_prior %>%
    gt()
prior class coef group resp dpar nlpar lb ub source
b default
b light_dark_c default
b log_mass_c default
b order_c default
b salinity_group9 default
b temp_c default
student_t(3, -2.7, 2.5) Intercept default
student_t(3, 0, 2.5) sd 0 default
sd id default
sd Intercept id default
b shape default
b order_c shape default
b salinity_group9 shape default
student_t(3, 0, 2.5) Intercept shape default

10.1.4 Run model

Here we run the model, I have hashed this out because I have saved the model for quick reloading.

⏭️ Skip this step if you have downloaded the saved model ‘mo2_mod_gamma.rds’

Here we reload the model

mo2_mod_gamma <- readRDS(file = paste0(output_mods_wd, "./mo2_mod_gamma.rds"))

10.1.5 Model diagnostics

Checking model convergence

color_scheme_set("red")
plot(mo2_mod_gamma, ask = F)

Checking rhat are equal to one

summary(mo2_mod_gamma)
##  Family: gamma 
##   Links: mu = log; shape = log 
## Formula: MO2 ~ temp_c + order_c + light_dark_c + log_mass_c + salinity_group + (1 | id) 
##          shape ~ salinity_group + order_c
##    Data: slope_tidy_smr (Number of observations: 893) 
##   Draws: 4 chains, each with iter = 8000; warmup = 1000; thin = 2;
##          total post-warmup draws = 14000
## 
## Multilevel Hyperparameters:
## ~id (Number of levels: 58) 
##               Estimate Est.Error l-95% CI u-95% CI Rhat Bulk_ESS Tail_ESS
## sd(Intercept)     0.31      0.03     0.26     0.38 1.00     4013     6644
## 
## Regression Coefficients:
##                       Estimate Est.Error l-95% CI u-95% CI Rhat Bulk_ESS
## Intercept                -2.54      0.06    -2.66    -2.43 1.00     2241
## shape_Intercept           2.58      0.07     2.45     2.72 1.00    12143
## temp_c                   -0.33      0.21    -0.75     0.09 1.00     6431
## order_c                   0.00      0.00    -0.00     0.01 1.00    11445
## light_dark_c              0.13      0.02     0.09     0.18 1.00    12311
## log_mass_c                0.81      0.10     0.61     1.01 1.00     2943
## salinity_group9          -0.10      0.08    -0.26     0.07 1.00     2304
## shape_salinity_group9     0.18      0.10    -0.03     0.37 1.00    12404
## shape_order_c            -0.04      0.01    -0.06    -0.02 1.00    11624
##                       Tail_ESS
## Intercept                 4183
## shape_Intercept          12331
## temp_c                    9409
## order_c                  12155
## light_dark_c             12266
## log_mass_c                5032
## salinity_group9           4394
## shape_salinity_group9    12389
## shape_order_c            12394
## 
## Draws were sampled using sampling(NUTS). For each parameter, Bulk_ESS
## and Tail_ESS are effective sample size measures, and Rhat is the potential
## scale reduction factor on split chains (at convergence, Rhat = 1).

Using leave one out (loo) measure of fit, the model appears to preform well, all, but one Pareto k estimates are good (k < 0.7)

loo(mo2_mod_gamma)
## 
## Computed from 14000 by 893 log-likelihood matrix.
## 
##          Estimate   SE
## elpd_loo   2256.8 38.9
## p_loo        71.6  7.1
## looic     -4513.5 77.8
## ------
## MCSE of elpd_loo is NA.
## MCSE and ESS estimates assume MCMC draws (r_eff in [0.7, 1.0]).
## 
## Pareto k diagnostic values:
##                          Count Pct.    Min. ESS
## (-Inf, 0.7]   (good)     892   99.9%   765     
##    (0.7, 1]   (bad)        1    0.1%   <NA>    
##    (1, Inf)   (very bad)   0    0.0%   <NA>    
## See help('pareto-k-diagnostic') for details.

Model predictions generally align with the observed data

color_scheme_set("red")
p <- pp_check(mo2_mod_gamma, type = "dens_overlay")
p

10.1.6 📈 Results

We did not see a meaningful difference between the routine metabolic rate for fish from the two salinity treatments.

10.1.6.1 Table S1

Table S1: Fixed effect Estimates (β) and 95% Credible Intervals (95% CI)

model_est <- fixef(mo2_mod_gamma, probs = c(0.025, 0.975)) %>%
    as.data.frame() %>%
    tibble::rownames_to_column(var = "Predictor") %>%
    dplyr::mutate(β = round(Estimate, 3), Q2.5 = round(Q2.5, 3), Q97.5 = round(Q97.5,
        3), `95% CI` = paste0("[", Q2.5, ", ", Q97.5, "]"))

model_est %>%
    dplyr::select(Predictor, "β", "95% CI") %>%
    gt()
Predictor β 95% CI
Intercept -2.545 [-2.659, -2.428]
shape_Intercept 2.585 [2.45, 2.715]
temp_c -0.335 [-0.755, 0.087]
order_c 0.002 [-0.002, 0.005]
light_dark_c 0.134 [0.086, 0.182]
log_mass_c 0.805 [0.609, 1.006]
salinity_group9 -0.096 [-0.259, 0.066]
shape_salinity_group9 0.177 [-0.025, 0.372]
shape_order_c -0.041 [-0.059, -0.023]

Looking at the marginal mean MO2 for each salinity treatment

em_results <- emmeans(mo2_mod_gamma, ~salinity_group)

em_results_df <- em_results %>%
    as.data.frame() %>%
    mutate(across(where(is.numeric), ~round(exp(.), 3)))

em_results_df %>%
    gt()
salinity_group emmean lower.HPD upper.HPD
0 0.079 0.070 0.088
9 0.071 0.064 0.080

These are the estimated mean constrast, we have exponentiated the log estimate to get the fold-change. This means the model predicts that sailianty group 0 has ~12% higher metabolic rate than group 9 (1.12)—but this difference is not credible, because the interval for that difference includes no difference (i.e. a ratio of 1.0).

contrast_results <- contrast(em_results, method = "pairwise")

contrast_results_df <- contrast_results %>%
    as.data.frame() %>%
    mutate(across(where(is.numeric), ~round(exp(.), 2)))

contrast_results_df %>%
    gt()
contrast estimate lower.HPD upper.HPD
salinity_group0 - salinity_group9 1.1 0.94 1.29

EEM contrasts for the two light phases, where -0.5 is dark, and 0.5 is light

em_results <- emmeans(mo2_mod_gamma, ~light_dark_c, at = list(light_dark_c = c(0.5,
    -0.5)))

contrast_results <- contrast(em_results, method = "pairwise", .rev = TRUE)

contrast_results_df <- contrast_results %>%
    as.data.frame() %>%
    mutate(across(where(is.numeric), ~round(exp(.), 2)))

contrast_results_df %>%
    gt()
contrast estimate lower.HPD upper.HPD
light_dark_c0.5 - (light_dark_c-0.5) 1.14 1.09 1.2

Here we we calculate the model-estimated marginal means for MO2 for a unit increase in mass and transform this onto a fold-change scale. Where log(mass) has be centered, a caule of 0 represents the mean mass of fish, and 1 corresponds to a fish with log(mass) = 1 unit above the mean (i.e. 2.7 times large).

log(mass2) = log(mass1)+1

If you exponentiate both sides:

mass2 = mass1 x exp(1)

mass2 ≈ mass1 x 2.718

em_mass <- emmeans(mo2_mod_gamma, ~log_mass_c, at = list(log_mass_c = c(0, 1)))

# Get the contrast (difference in log MO₂), then back-transform
contrast(em_mass, method = "revpairwise") %>%
    as.data.frame() %>%
    mutate(across(where(is.numeric), exp)) %>%
    mutate(across(where(is.numeric), ~round(., 2))) %>%
    gt()
contrast estimate lower.HPD upper.HPD
log_mass_c1 - log_mass_c0 2.23 1.86 2.76

Pulling the emmeans draws for our plot

emmeans_draws_rmr <- mo2_mod_gamma %>%
    emmeans(~salinity_group) %>%
    gather_emmeans_draws() %>%
    dplyr::mutate(.value = exp(.value), salinity_group = as.character(salinity_group))

emmeans_contrast_draws_rmr <- mo2_mod_gamma %>%
    emmeans(~salinity_group) %>%
    contrast(method = "pairwise") %>%
    gather_emmeans_draws() %>%
    dplyr::mutate(.value = exp(.value))

10.1.6.2 Figure S7 (Figure 1a)

NOTE: This plot is in the main text of the manuscript as Figure 1a

fig_1a <- ggplot() +
  geom_violin(data = slope_tidy_smr,
              aes(x = salinity_group, y = MO2, fill = salinity_group),
              color = NA, alpha = 0.2) +
  geom_jitter(data = slope_tidy_smr,
              aes(x = salinity_group, y = MO2, fill = salinity_group),
              shape = 21, width = 0.3, size = 1, color = "black", alpha = 0.1) +
  geom_point(data = mean_mo2_salinity,
             aes(x = salinity_group, y = mean_mo2, fill = salinity_group),
             size = 4, alpha = 1, stroke = 2, color = "black", shape = 21,
             position = position_nudge(x = 0.05)) +
  stat_pointinterval(data = emmeans_draws_rmr, 
                     aes(x = salinity_group, y = .value),
                     color = "black", fill = "grey", point_interval = "mean_qi", .width = 0.95, shape = 21,  stroke = 2, point_size = 4, alpha = 1,
                     position = position_nudge(x = -0.05)) +
  scale_y_continuous(limits = c(0.00, 0.35)) +  # 👈 Set y-axis range here
  scale_fill_manual(values = c("#4B5320", "#000080")) +
  scale_colour_manual(values = c("#4B5320", "#000080")) +
  theme(
    legend.position = "none",
    panel.grid.major = element_blank(),
    panel.grid.minor = element_blank(),
    panel.border = element_rect(colour = "black", fill = NA, linewidth = 1),
    panel.background = element_rect(fill = "white", colour = NA),
    plot.background = element_rect(fill = "white", colour = NA)
  ) +
  labs(
    subtitle = "",
    x = "Salinity group (ppt)",
    y = "Routine MO2 (mg O2 g/h)"
  )

fig_1a

Figure 1a: Routine metabolic rate (i.e. MO2 (mg-1 O2 h-1) measured during SMR meassurments) plotted by salinity treatment. The small transparent points are the observed values, the shaded area behind the points is the a kernel density of the observed data, the large coloured point (to the right) is the observed mean, the large grey point with error bars (to the left) is the model estimated marginal mean (eemean) 95% Credible Intervals (95% CI).

# ggsave(filename = paste0(output_fig_wd, './ms_figures./figure-1a.pdf'), plot
# = fig_1a, width = 210/2, height = 297/2, # Specify the height of the plot
# units = 'mm')

10.2 SMR

10.2.1 Formating and scaling data

Here we are filtering the data frame to have only measure per fish for the SMR estimate

scale_list <- c("temp_mean", "log_mass", "cycles")

smr <- slope_tidy_smr %>%
    dplyr::group_by(id) %>%
    dplyr::reframe(temp_mean = mean(temp), log_mass = log_mass[1], mass = mass[1],
        SMR = SMR[1], salinity_group = salinity_group[1], cycles = length(order)) %>%
    dplyr::mutate(across(all_of(scale_list), ~scale(.x, center = TRUE, scale = FALSE),
        .names = "{.col}_c"))

Summary of predictors used in the model below

smr %>%
    dplyr::reframe(temp_range = paste0(round(min(temp_mean, na.rm = TRUE), 2), "–",
        round(max(temp_mean, na.rm = TRUE), 2)), mass_range = paste0(round(min(mass,
        na.rm = TRUE), 2), "–", round(max(mass, na.rm = TRUE), 2)), cycle_range = paste0(round(min(cycles,
        na.rm = TRUE), 2), "–", round(max(cycles, na.rm = TRUE), 2)), number_fish = length(unique(id))) %>%
    gt()
temp_range mass_range cycle_range number_fish
13.87–14.26 0.21–1.6 10–23 58

10.2.2 Model structure

Here we will use a Bayesian Generalised Linear Mixed Model (GLMM) with a Gamma distribution and a log link Gamma(link = "log"), where the shape parameter (α) is also modelled as a function of the salinity group, shape ~ salinity_group. This models MO2 by salinity during the SMR phase to see if the fish held at different salinities have different SMRs. We have also added a few scaled predictors, that may help describe variation in the data, such as fish mass (g; 0.21–1.6) mean temperature (°C; 13.841–14.277), and the number of cycles over which SMR was estimated (1–23). We allowed the the shape parameter (α) to vary as a function of salinity_group to improve fit.

smr_gamma_bf <- bf(SMR ~ temp_mean_c + cycles_c + log_mass_c + salinity_group, shape ~
    salinity_group, family = Gamma(link = "log"))

10.2.3 Prior selection

These are the default priors for the model. We will use these.

priors_default <- get_prior(smr_gamma_bf, data = smr, family = Gamma(link = "log"))
priors_default %>%
    gt()
prior class coef group resp dpar nlpar lb ub source
b default
b cycles_c default
b log_mass_c default
b salinity_group9 default
b temp_mean_c default
student_t(3, -2.8, 2.5) Intercept default
b shape default
b salinity_group9 shape default
student_t(3, 0, 2.5) Intercept shape default

10.2.4 Run model

Here we run the model, I have hashed this out because I have saved the model for quick reloading.

⏭️ Skip this step if you have downloaded the saved model ‘smr_mod_gamma.rds’

Here we reload the model

smr_mod_gamma <- readRDS(file = paste0(output_mods_wd, "./smr_mod_gamma.rds"))

10.2.5 Model diagnostics

Checking model convergence

color_scheme_set("red")
plot(smr_mod_gamma, ask = F)

Using leave one out (loo) measure of fit, the model appears to preform well, two Pareto k estimates falls outside the good range (0.7, 1]

loo(smr_mod_gamma)
## 
## Computed from 14000 by 58 log-likelihood matrix.
## 
##          Estimate   SE
## elpd_loo    134.6 11.0
## p_loo        11.0  4.8
## looic      -269.2 22.0
## ------
## MCSE of elpd_loo is NA.
## MCSE and ESS estimates assume MCMC draws (r_eff in [0.8, 0.9]).
## 
## Pareto k diagnostic values:
##                          Count Pct.    Min. ESS
## (-Inf, 0.7]   (good)     56    96.6%   4028    
##    (0.7, 1]   (bad)       1     1.7%   <NA>    
##    (1, Inf)   (very bad)  1     1.7%   <NA>    
## See help('pareto-k-diagnostic') for details.

Model predictions generally align with the observed data, but there is a lot of uncertainty around this estimate.

color_scheme_set("red")
p <- pp_check(smr_mod_gamma, type = "dens_overlay")
p

10.2.6 📈 Results

We did not see a meaningful difference between the routine metabolic rate for fish from the two salinity treatments.

10.2.6.1 Table S2

Table S2: Fixed effect Estimates (β) and 95% Credible Intervals (95% CI) from a Bayesian Generalised Linear Mixed Model (GLMM) with a Gamma distribution and a log link

model_est <- fixef(smr_mod_gamma, probs = c(0.025, 0.975)) %>%
    as.data.frame() %>%
    tibble::rownames_to_column(var = "Predictor") %>%
    dplyr::mutate(β = round(Estimate, 3), Q2.5 = round(Q2.5, 3), Q97.5 = round(Q97.5,
        3), `95% CI` = paste0("[", Q2.5, ", ", Q97.5, "]"))

model_est %>%
    dplyr::select(Predictor, "β", "95% CI") %>%
    gt()
Predictor β 95% CI
Intercept -2.740 [-2.879, -2.603]
shape_Intercept 2.010 [1.428, 2.532]
temp_mean_c -0.364 [-1.51, 0.802]
cycles_c -0.005 [-0.026, 0.018]
log_mass_c 0.816 [0.597, 1.034]
salinity_group9 -0.086 [-0.283, 0.111]
shape_salinity_group9 0.117 [-0.733, 0.945]

Looking at the marginal mean difference between salinity groups

em_results <- emmeans(smr_mod_gamma, ~salinity_group)

em_results_df <- em_results %>%
    as.data.frame() %>%
    mutate(across(where(is.numeric), ~round(exp(.), 3)))


em_results_df %>%
    gt()
salinity_group emmean lower.HPD upper.HPD
0 0.065 0.056 0.074
9 0.059 0.052 0.067
contrast_results_df <- contrast_results %>%
    as.data.frame() %>%
    mutate(across(where(is.numeric), ~exp(.)))

contrast(em_results, method = "pairwise") %>%
    as.data.frame() %>%
    mutate(across(where(is.numeric), ~round(exp(.), 2))) %>%
    gt()
contrast estimate lower.HPD upper.HPD
salinity_group0 - salinity_group9 1.09 0.89 1.32

Here we we calculate the model-estimated marginal means for SMR for a unit increase in mass and transform this onto a fold-change scale. Where log(mass) has be centered, a vaule of 0 represents the mean mass of fish, and 1 corresponds to a fish with log(mass) = 1 unit above the mean (i.e. 2.7 times large).

em_mass <- emmeans(smr_mod_gamma, ~log_mass_c, at = list(log_mass_c = c(0, 1)))

# Get the contrast (difference in log MO₂), then back-transform
contrast(em_mass, method = "revpairwise") %>%
    as.data.frame() %>%
    mutate(across(where(is.numeric), exp)) %>%
    mutate(across(where(is.numeric), ~round(., 2))) %>%
    gt()
contrast estimate lower.HPD upper.HPD
log_mass_c1 - log_mass_c0 2.26 1.84 2.84

Pulling the emmeans draws for our plot

emmeans_draw_smr <- smr_mod_gamma %>%
    emmeans(~salinity_group) %>%
    gather_emmeans_draws() %>%
    dplyr::mutate(.value = exp(.value), salinity_group = as.character(salinity_group))

emmeans_contrast_draws_smr <- smr_mod_gamma %>%
    emmeans(~salinity_group) %>%
    contrast(method = "pairwise") %>%
    gather_emmeans_draws() %>%
    dplyr::mutate(.value = exp(.value))

10.2.6.2 Figure S8 (Figure 1b)

This plot is in the main text of the manuscript as Figure 1b

mean_smr_salinity <- smr %>%
  dplyr::group_by(salinity_group) %>%
  dplyr::reframe(mean_SMR = mean(SMR, na.rm = TRUE))


fig_1b <- ggplot() +
  geom_violin(data = smr,
              aes(x = salinity_group, y = SMR, fill = salinity_group),
              color = NA, alpha = 0.2) +
  geom_jitter(data = smr,
              aes(x = salinity_group, y = SMR, fill = salinity_group),
              shape = 21, width = 0.3, size = 1, color = "black", alpha = 0.1) +
  geom_point(data = mean_smr_salinity,
             aes(x = salinity_group, y = mean_SMR, fill = salinity_group),
             size = 4, alpha = 1, stroke = 2, color = "black", shape = 21,
             position = position_nudge(x = 0.05)) +
  stat_pointinterval(data = emmeans_draw_smr,
                     aes(x = salinity_group, y = .value),
                     color = "black", fill = "grey", point_interval = "mean_qi", .width = 0.95, shape = 21,  stroke = 2, point_size = 4, alpha = 1,
                     position = position_nudge(x = -0.05)) +
  scale_y_continuous(limits = c(0.00, 0.35)) +
  scale_fill_manual(values = c("#4B5320", "#000080")) +  # Custom fill colours
  scale_colour_manual(values = c("#4B5320", "#000080")) +
  theme(
    legend.position = "none",
    panel.grid.major = element_blank(),
    panel.grid.minor = element_blank(),
    panel.border = element_rect(colour = "black", fill = NA, linewidth = 1),
    panel.background = element_rect(fill = "white", colour = NA),
    plot.background = element_rect(fill = "white", colour = NA)
  ) +
  labs(
    subtitle = "",
    x = "Salinity group (ppt)",
    y = "Standard metabolic rate (SMR; mg O2 g/h)"
  )

fig_1b

labs( y = expression(“mg O”[2]~“h”^-1), x = “Dissolved oxygen (DO; % saturation)” ) + Figure 1b: The standard metabolic rate estimate (mg-1 O2 h-1) plotted by salinity treatment. The small transparent points are the observed values, the shaded area behind the points is the a kernel density of the observed data, the large coloured point (to the right) is the observed mean, the large grey point with error bars (to the left) is the model estimated marginal mean (eemean) 95% Credible Intervals (95% CI).

# ggsave(filename = paste0(output_fig_wd, './ms_figures./figure-1b.pdf'), plot
# = fig_1b, width = 210/2, height = 297/2, # Specify the height of the plot
# units = 'mm')

10.3 Incremental regression analyses

Here we are following the methods Urbina et al. (2012)[1] with an incremental regression analyses, in order to determine the best fit for the MO2 vs O2 data

This analysis approach evaluates the relative ‘fit’ of each polynomial order equation starting at zero and increasing to the third order, permitting a mathematical assessment of whether fish were oxyconforming or oxyregulatoring. If the data is best fitted/predicted by a single linear relationship (1st-order polynomial) with a positive slope, this would suggests the fish were oxyconforming. Alternately, if the relationship is best modelled by a flat regression (0th-order polynomial), or a higher order polynomial (2nd or 3rd-order polynomial) the fish is likely oxyregulatoring.

10.3.1 Building Bayesian regressions

Here we are using a Bayesian approach to model fitting with brm. These models take a long time to run, so I have saved them and re-loaded them to save time. I have also saved the summary data produced from the models, to save time, you can simply skip the hashed code and input the resulting summary data.

We will run our custom function, bayes_incremental_regression_by_id.

Let’s make define the output directory

output_mods_bayes_wd <- paste0(wd, "./output-mod./bayes-regs")

⏭️ Skip this step if you have already run this once, or have downloaded the saved models or saved data files from GitHub (that’s why its hashed out). ⌛This code takes a while to run

During this part of the script I received an error, ‘Caused by error in socketConnection():’. I think the system may be hitting a limit on the number of simultaneous socket connections. If you this part of the code and also get this issue, try reducing the number of parallel workers: plan(multisession, workers = 2), or run again after the error.

# ids <- slope_tidy %>% dplyr::distinct(id) %>% pull(id) plan(multisession)
# future_map( ids, bayes_incremental_regression_by_id, id_name = 'id', data =
# slope_tidy, response = 'MO2_g', predictor = 'DO', seed_number = 143019,
# save_models = TRUE, mod_output_wd = output_mods_bayes_wd ) plan(sequential)

Load all models and store in a list, will use a lot of memory. You can also skip this step and load the resulting data frames below. I am using the custom function load_rds, so we can compare them and generate predictions.

⏭️ Skip this step if you have downloaded the saved data pulled from these models have also hased this out, bayes_reg_mods_fit.csv’ and ‘bayes_reg_mods_predictions.csv’. These are in the mod-data directory.

# bayes_reg_mods <- load_rds(model_dw = output_mods_bayes_wd)

10.3.2 Model fits

⏭️ Skip this step if you have downloaded 💿 bayes_reg_mods_fit.csv.

It gets model fit parameters loo and r2 using the custom function, incremental_regression_bayes_fits. ⌛ This code takes a while to run ⌛

# bayes_reg_mods_fit <- incremental_regression_bayes_fits(models =
# bayes_reg_mods) write.csv(bayes_reg_mods_fit, paste0(mod_data_wd,
# './bayes_reg_mods_fit.csv'), row.names = FALSE)

Reading in this model fit data frame, in the case you did not load in all the models.

bayes_reg_mods_fit <- read.csv(paste0(mod_data_wd, "./bayes_reg_mods_fit.csv"))

Selecting the best fitting model

elpd_loo, or the expected log pointwise predictive density for leave-one-out cross-validation, is a metric used in Bayesian model evaluation to assess the predictive accuracy of a model. The elpd_loo is an approximation of how well the model is expected to predict new data, based on leave-one-out cross-validation. Higher elpd_loo values indicate better predictive performance.

best_fit_bayes_reg <- bayes_reg_mods_fit %>%
    dplyr::group_by(id) %>%
    dplyr::mutate(elpd_loo_rank = rank(-elpd_loo)) %>%
    dplyr::select(id, model_type, elpd_loo, r2, elpd_loo_rank, r2_q2.5, r2_q97.5,
        estimate_DO, conf.low_DO, conf.high_DO) %>%
    ungroup()

10.3.3 Model predictions

⏭️ Skip this step if you have downloaded 💿 bayes_reg_mods_predictions.csv.

It pulls our model predictions using a custom function bayes_mod_predictions.

# bayes_reg_mods_predictions <- bayes_mod_predictions(models = bayes_reg_mods,
# original_data = slope_tidy) write.csv(bayes_reg_mods_predictions,
# paste0(mod_data_wd, './bayes_reg_mods_predictions.csv'), row.names = FALSE)

Reading in the predicted data

bayes_reg_mods_predictions <- read.csv(paste0(mod_data_wd, "./bayes_reg_mods_predictions.csv"))

We are going to combined this with our best fitting model df, so we know how they ranks for LOO.

bayes_reg_mods_predictions <- left_join(bayes_reg_mods_predictions, best_fit_bayes_reg,
    by = c("id", "model_type"))

10.3.4 📈 Results

10.3.4.1 Model selection summary

In most cases, the relationship between MO2 and DO was best modelled with a 2nd-order polynomial (n = 22, 38% of fish), followed by a 3rd-order polynomial (n = 15, 26%), 1st-order polynomial (n = 11, 19%), and finally 0th-order polynomial (n = 10, 17%).

For the two most common models, 2nd- and 3rd-order polynomials, this could suggest the presence of a critical oxygen threshold (Pcrit) where the relationship between MO2 and O2 changes. To confirm the presence of a Pcrit, we need to validated the shape of the polynomials (see Pcrit model below). In any case, these type of models are indicative of oxyregulator.

The next most common model was is 1st-order polynomial. In the case of the 1st-order polynomials, it suggest the presences of linear relationship between o2 and MO2, which is indicative of oxyconformer. However, to be true evidence of a oxyconformer this relationship should be positive (i.e. as O2 falls MO2 also falls). 13 of the 18 individuals best modelled with a linear function had positive estimates with credible intervals that did not overlap with zero (Table S3).

Lastly, 0th-order was the least common (n = 3, 5%), it suggests that MO2 does not show a statistically significant dependence on the O2. In other words, the metabolic rate does not adjust based on oxygen availability, and there is no clear critical oxygen threshold (Pcrit) where the relationship changes. This is indicative of a oxyregulator.

best_mod <- best_fit_bayes_reg %>%
    dplyr::filter(elpd_loo_rank == 1)

total_fish <- nrow(best_mod)

mod_summary_table <- best_mod %>%
    dplyr::group_by(model_type) %>%
    dplyr::reframe(n = length(id), percent = round((n/total_fish) * 100, 2)) %>%
    dplyr::mutate(best_model_name = case_when(model_type == "lm_0" ~ "0th-order polynomial",
        model_type == "lm_1" ~ "1st-order polynomial", model_type == "lm_2" ~ "2nd-order polynomial",
        model_type == "lm_3" ~ "3rd-order polynomial", TRUE ~ "ERROR"))


table_bwm <- mod_summary_table %>%
    dplyr::select(best_model_name, everything(), -model_type) %>%
    gt() %>%
    cols_align(align = "center", columns = everything())

table_bwm
best_model_name n percent
0th-order polynomial 10 17.24
1st-order polynomial 11 18.97
2nd-order polynomial 22 37.93
3rd-order polynomial 15 25.86

Summary of fish best model with a linear function.

10.3.5 Table S3

Table S3: Ten fish that were best modelled with a linear function, showing r2, and estimate (). Only two fish had positive estimates with credible intervals that did not overlap with zero, which are highlighted as conforming in the table. Thus in total, 13 of 58 fish showed sufficient evidence to be conforming.

table_lm_1 <- best_mod %>%
    dplyr::filter(model_type == "lm_1") %>%
    dplyr::mutate(r_sq_ci = paste0(format(round(r2, 3), scientific = FALSE), " (",
        format(round(r2_q2.5, 3), scientific = FALSE), " to ", format(round(r2_q97.5,
            3), scientific = FALSE), ")"), est_ci = paste0(format(round(estimate_DO,
        4), scientific = FALSE), " (", format(round(conf.low_DO, 4), scientific = FALSE),
        " to ", format(round(conf.high_DO, 4), scientific = FALSE), ")"), conformer = if_else(conf.low_DO >
        0, "Conforming", "Not conforming")) %>%
    dplyr::select(id, estimate_DO, r_sq_ci, est_ci, conformer) %>%
    dplyr::arrange(conformer, desc(est_ci)) %>%
    dplyr::ungroup()

first_order_ids <- table_lm_1 %>%
    distinct(id) %>%
    pull(.)

mean_mo2 <- slope_tidy %>%
    dplyr::filter(id %in% first_order_ids) %>%
    dplyr::group_by(id) %>%
    dplyr::reframe(mean_MO2_g = mean(MO2_g, na.rm = TRUE))

table_lm_1 <- left_join(table_lm_1, mean_mo2, by = "id")

table_lm_1 <- table_lm_1 %>%
    dplyr::mutate(percent_change = (estimate_DO/mean_MO2_g) * 100) %>%
    dplyr::select(id, r_sq_ci, est_ci, percent_change, conformer)


table_lm_1 %>%
    gt() %>%
    cols_align(align = "center", columns = everything()) %>%
    cols_label(id = "Fish ID", r_sq_ci = "r2 (CI)", est_ci = "Estimate (CI)", percent_change = "Percentage change",
        conformer = "Evidence of oxyconforming")
Fish ID r2 (CI) Estimate (CI) Percentage change Evidence of oxyconforming
d_9_25nov_3 0.197 (0.012 to 0.394) 0.0011 ( 0.0003 to 0.0019) 0.7437151 Conforming
a_9_22nov_4 0.312 (0.094 to 0.491) 0.0008 ( 0.0004 to 0.0013) 0.5739880 Conforming
b_9_22nov_1 0.231 (0.055 to 0.396) 0.0005 ( 0.0002 to 0.0008) 0.4310066 Conforming
d_9_25nov_2 0.330 (0.102 to 0.507) -0.0009 (-0.0013 to -0.0004) -0.7599381 Not conforming
a_0_25nov_1 0.137 (0.002 to 0.331) -0.0004 (-0.0007 to 0.0000) -0.2788575 Not conforming
c_0_22nov_3 0.072 (0.000 to 0.256) 0.0005 (-0.0005 to 0.0016) 0.3013443 Not conforming
a_0_24nov_3 0.058 (0.000 to 0.202) 0.0005 (-0.0003 to 0.0012) 0.2832654 Not conforming
c_9_27nov_2 0.068 (0.000 to 0.255) 0.0003 (-0.0003 to 0.0008) 0.1697994 Not conforming
a_9_22nov_1 0.084 (0.000 to 0.266) 0.0003 (-0.0002 to 0.0008) 0.2272965 Not conforming
b_0_25nov_2 0.108 (0.001 to 0.290) 0.0003 ( 0.0000 to 0.0006) 0.2275231 Not conforming
a_0_24nov_1 0.029 (0.000 to 0.134) 0.0002 (-0.0005 to 0.0009) 0.1072286 Not conforming

10.3.5.1 Ploting all models

Data set with all slopes and which model was best

best_fit <- left_join(slope_tidy, best_mod, by = "id")

Saving all regression, and highlighting the model that has the best fit, based on AIC values. This is called combined_reg_plots.pdf

# # Create a list to store the plots plots <- list() model_preds_list <- list()
# incremental_reg_bayes_wd <- paste0(output_fig_wd,
# './incremental_regressions') for (id_i in ids) { # Filter data for the
# current ID df_i <- bayes_reg_mods_predictions %>% dplyr::filter(id == id_i)
# %>% dplyr::mutate(line_size = if_else(elpd_loo_rank == 1, 2, 1), alpha_value
# = if_else(elpd_loo_rank == 1, 1, 0.4)) x_min <- df_i %>% dplyr::reframe(min =
# min(DO), na.rm = TRUE) %>% dplyr::pull(min) y_max <- df_i %>%
# dplyr::reframe(max = max(MO2_g), na.rm = TRUE) %>% dplyr::pull(max)
# best_weighted_model_i <- best_fit_bayes_reg %>% dplyr::filter(id == id_i &
# elpd_loo_rank == 1) poly_i_name <- best_weighted_model_i %>%
# dplyr::mutate(name = case_when( model_type == 'lm_0' ~ '0th-order',
# model_type == 'lm_1' ~ '1st-order', model_type == 'lm_2' ~ '2nd-order',
# model_type == 'lm_3' ~ '3rd-order', TRUE ~ 'ERROR' )) %>% dplyr::pull(name)
# r_i <- best_weighted_model_i %>% dplyr::mutate(r_sq_ci = paste0(round(r2, 3),
# ' (', round(r2_q2.5, 3), '–', round(r2_q97.5, 3), ')')) %>%
# dplyr::pull(r_sq_ci) # Create the plot plot_i <- ggplot() + geom_ribbon(data
# = df_i, aes(x = DO, y = predicted, ymin = pred_lower, ymax = pred_upper, fill
# = model_type), alpha = 0.1) + geom_line(data = df_i, aes(x = DO, y =
# predicted, colour = model_type, linewidth = line_size, alpha = alpha_value))
# + geom_point(data = df_i %>% dplyr::filter(elpd_loo_rank == 1), aes(x = DO, y
# = MO2_g), alpha = 0.6, colour = 'black', size = 2) +
# scale_colour_manual(values = c('red', 'blue', 'green', 'purple'), labels =
# c('0th Order', '1st Order', '2nd Order', '3rd Order')) +
# scale_size_identity() + # Use the size values directly
# scale_alpha_identity(guide = 'none') + # Remove the alpha legend
# annotate('text', x = x_min, y = y_max, label = paste0('Best fit:
# ',poly_i_name, '\n', 'r2 = ', r_i), hjust = 0, vjust = 1, size = 4) + labs(
# title = paste(id_i), x = 'Dissolved oxygen percentage (DO)', y = 'MO2 (o2
# mg/g/h)', colour = 'Model') + theme_classic() + theme(legend.position =
# 'none') # Store the plot plots[[id_i]] <- plot_i print(plot_i) } pdf(file =
# paste0(incremental_reg_bayes_wd, './combined_reg_plots.pdf'), width = 8,
# height = 6) dev.off() # Close the PDF device

Getting a plot for each best fit regression, and overlaying a global model for that model type.

Here we are grouping fish by best fitting model and getting an average trend. I have hashed the code so you don’t need to re-run the models

⏭️ Skip this step if you have downloaded the global models ‘.rds’ in the ‘bayes-regs-global’ folder

# seed_number = 143019 # Set up parallel backend future::plan(multisession,
# workers = 4) # Adjust workers based on system resources # Define model
# formulas and data model_formulas <- list( lm_0 = bf(MO2_g ~ 1, family =
# gaussian()), lm_1 = bf(MO2_g ~ DO, family = gaussian()), lm_2 = bf(MO2_g ~
# poly(DO, 2), family = gaussian()), lm_3 = bf(MO2_g ~ poly(DO, 3), family =
# gaussian()) ) model_data <- list( lm_0 = best_fit %>% filter(model_type ==
# 'lm_0'), lm_1 = best_fit %>% filter(model_type == 'lm_1'), lm_2 = best_fit
# %>% filter(model_type == 'lm_2'), lm_3 = best_fit %>% filter(model_type ==
# 'lm_3') ) # Run models in parallel with future_map() models <-
# future_map(names(model_formulas), ~{ brm( formula = model_formulas[[.x]],
# data = model_data[[.x]], cores = 4, seed = seed_number, save_pars =
# save_pars(all = TRUE), sample_prior = FALSE, silent = TRUE, file =
# paste0(output_mods_bayes_global_wd, './', .x, '_global') ) }) # Assign model
# names to results names(models) <- names(model_formulas) # Stop parallel plan
# after execution future::plan(sequential)

⏭️ Skip this step if you have downloaded the global_models_pred_df.csv

# global_models_preds <- list() for (mode_name in names(bayes_reg_mods_global))
# { mod_i <- bayes_reg_mods_global[[mode_name]] mode_type =
# str_remove(mode_name, '_global') model_predictions_i <-
# as.data.frame(fitted(mod_i, summary = TRUE)) %>% dplyr::mutate(model =
# mode_name, model_type = mode_type) %>% dplyr::rename(pred_lower = Q2.5,
# pred_upper = Q97.5, predicted = Estimate, pred_error = Est.Error) %>%
# dplyr::select(model, everything()) original_data_i <- best_fit %>%
# filter(model_type == mode_type) %>% dplyr::select(-model_type)
# model_predictions_original_i <- cbind(model_predictions_i, original_data_i)
# global_models_preds[[mode_name]] <- model_predictions_original_i }
# global_models_pred_df <- bind_rows(global_models_preds)
# write.csv(global_models_pred_df, paste0(mod_data_wd,
# '/.global_models_pred_df.csv'), row.names = FALSE)

Load data frame

global_models_pred_df <- read.csv(paste0(mod_data_wd, "/.global_models_pred_df.csv"))

10.3.5.2 Figure S9

Best fit regressions with global models based on that polynomial order.

global_models_pred_df <- global_models_pred_df %>%
  dplyr::mutate(model_name = case_when(
      model_type == "lm_0" ~ "0th-order polynomial",
      model_type == "lm_1" ~ "1st-order polynomial",
      model_type == "lm_2" ~ "2nd-order polynomial",
      model_type == "lm_3" ~ "3rd-order polynomial",
      TRUE ~ "ERROR"
    ))

annotation_data <- mod_summary_table %>%
  dplyr::select(model_type, best_model_name, n)

bayes_reg_mods_predictions_best <- bayes_reg_mods_predictions %>% 
  dplyr::filter(elpd_loo_rank == 1) 

fig_2 <- ggplot() +
  geom_line(data = bayes_reg_mods_predictions_best,
            aes(x = DO, y = predicted, color = id), size = 1) +
  # geom_smooth(data = best_fit,
  #           aes(x = DO, y = MO2_g, color = id), size = 1, alpha = 1, se = FALSE) +
  geom_ribbon(data = global_models_pred_df,
              aes(x = DO, ymin = pred_lower
, ymax = pred_upper, group = model),
              fill = "#FC6C85", alpha = 0.2) +  # Shaded confidence intervals
  geom_line(data = global_models_pred_df,
            aes(x = DO, y = predicted), linewidth = 1.5, color = "#FF007F") +
  facet_wrap(~model_type) +
  scale_color_grey(start = 0.1, end = 0.9) +
  labs(
      title = paste("Best fit regressions grouped by polynomial order"),
      x = "Dissolved oxygen percentage (DO)",
      y = "MO2 (O2 mg/g/h)") +
  theme_classic() +
  theme(legend.position = "none") +
  geom_text(data = annotation_data,
            aes(x = -Inf, y = Inf, label = paste0("italic(n) == ", n)),
            hjust = -0.1, vjust = 1.2, inherit.aes = FALSE, parse = TRUE)
fig_2

Figure S9: Individual regression curves showing the relationship between mass-specific oxygen consumption rate (ṀO₂; mg⁻¹ O₂ h⁻¹) and ambient dissolved oxygen (DO; % saturation), grouped by the best-fitting polynomial order (0ᵗʰ to 3ʳᵈ). Each panel represents fish whose ṀO₂–DO relationship was best modelled by that polynomial order, based on leave-one-out cross-validation (LOO-CV). Grey lines represent individual model fits; the bold pink line shows the group-level trend within each polynomial class. Fish were most commonly best fit by a 2ⁿᵈ-order polynomial (n = 22), followed by 3ʳᵈ-order (n = 15), 1ˢᵗ-order (n = 11), and 0ᵗʰ-order (**n = 10). Higher-order fits (2ⁿᵈ and 3ʳᵈ) and 0ᵗʰ-order fits are generally indicative of oxyregulating responses, while 1ˢᵗ-order fits may suggest oxyconforming behaviour when positively sloped.

# ggsave(filename = paste0(output_fig_wd, './ms_figures./figure-2.pdf'), plot =
# fig_2, width = 210, height = 297/2, # Specify the height of the plot units =
# 'mm')

10.4 O₂crit model

We will calculate O₂crit using Chabot method and function calcO2crit.

This function uses the fifth percentile of the MO2 values observed at dissolved oxygen levels ≥ 80% air saturation as the criterion to assess low MO2 values. The algorithm then identifies all the MO2 measurements greater than this minimally acceptable MO2 value. Within this sub-set, it identifies the MO22 measurement made at the lowest DO and thereafter considers this DO as candidate for breakpoint (named pivotDO in the script). A regression is then calculated using observations at DO levels < pivotDO, and a first estimate of O2crit is calculated as the intersection of this regression line with the horizontal line representing SMR. The script then goes through validation steps to ensure that the slope of the regression is not so low that the line, projected to normoxic DO levels, passes below any MO2 values observed in normoxia. It also ensures that the intercept is not greater than zero. Corrective measures are taken if such problems are encountered.

lowestMO2 default is the quantile(MO2[DO >= 80], p=0.05). It is used to segment the data and locate the pivotDO.

ids <- slope_tidy %>%
    dplyr::distinct(id) %>%
    dplyr::pull()

pcrit_model_df_list <- list()
pcrit_models <- list()

for (id_i in ids) {

    df_i <- slope_tidy %>%
        dplyr::filter(id == id_i)

    o2crit <- calcO2crit(Data = df_i, SMR = df_i$SMR[1], lowestMO2 = NA, gapLimit = 4,
        max.nb.MO2.for.reg = 7)

    best_mod_type <- best_mod %>%
        dplyr::filter(id == id_i) %>%
        pull(model_type)

    vaule <- o2crit$o2crit
    lowestMO2 = quantile(df_i$MO2[df_i$DO >= 80], p = 0.05)
    SMR <- o2crit$SMR
    nb_mo2_conforming <- o2crit$Nb_MO2_conforming
    r2 <- o2crit$r2
    method <- o2crit$Method
    p <- o2crit$P[1]
    lethal_point <- min(o2crit$lethalPoints)

    pcrit_model_df <- tibble(id = id_i, pcrit_vaule = vaule, pcrit_smr = SMR, pcrit_lowestMO2 = lowestMO2,
        pcrit_nb_mo2_conforming = nb_mo2_conforming, pcrit_lethal_points = lethal_point,
        pcrit_r2 = r2, pcrit_method = method, pcrit_p = p, best_mod_type = best_mod_type)

    pcrit_model_df_list[[id_i]] <- pcrit_model_df

    pcrit_models[[id_i]] <- o2crit

}

pcrit_model_df <- bind_rows(pcrit_model_df_list)

10.4.1 Ploting O2crit

Here’s we are saving all O2crit models, even those MO2-O2 relationships best modelled by 0th- and 1st-order polynomials.

# # Create output directory if needed output_fig_pcrit_chabot_wd <-
# file.path(output_fig_wd, 'model_chabot') if
# (!dir.exists(output_fig_pcrit_chabot_wd)) {
# dir.create(output_fig_pcrit_chabot_wd) } ids <- slope_tidy %>%
# dplyr::distinct(id) %>% dplyr::pull() pcrit_chabot_list <- list() # Open a
# single PDF device pdf(file = file.path(output_fig_pcrit_chabot_wd,
# 'combined_chabot_plots.pdf'), width = 8, height = 6) for (id_i in ids) {
# best_mod_type <- best_mod %>% dplyr::filter(id == id_i) %>% pull(model_type)
# r2 <- pcrit_model_df %>% dplyr::filter(id == id_i) %>% dplyr::mutate(pcrit_r2
# = round(pcrit_r2, 3)) %>% dplyr::pull(pcrit_r2) conforming <- pcrit_model_df
# %>% dplyr::filter(id == id_i) %>% dplyr::mutate(pcrit_nb_mo2_conforming =
# round(pcrit_nb_mo2_conforming, 3)) %>% dplyr::pull(pcrit_nb_mo2_conforming) P
# <- pcrit_model_df %>% dplyr::filter(id == id_i) %>% dplyr::mutate(pcrit_p =
# round(pcrit_p, 3)) %>% dplyr::pull(pcrit_p) SMR <- pcrit_model_df %>%
# dplyr::filter(id == id_i) %>% dplyr::mutate(pcrit_smr = round(pcrit_smr, 3))
# %>% dplyr::pull(pcrit_smr) lowestMO2 <- pcrit_model_df %>% dplyr::filter(id
# == id_i) %>% dplyr::mutate(pcrit_lowestMO2 = round(pcrit_lowestMO2, 3)) %>%
# dplyr::pull(pcrit_lowestMO2) # Generate and render the plot
# plotO2crit(o2critobj = pcrit_models[[id_i]]) # Add a title mtext( text =
# paste0(id_i), side = 3, line = 2, adj = 0, col = 'blue', font = 2, cex = 1.2
# ) mtext( text = paste0('R2 = ', r2, '; p = ', P, '; CP < SMR = ', conforming,
# '; SMR = ', SMR, '; lowestMO2 = ',lowestMO2), side = 3, line = 1, adj = 0,
# col = 'blue', font = 1, cex = 0.8 ) mtext( text = paste0('Best model fit = ',
# best_mod_type), side = 3, line = 0, adj = 0, col = 'red', font = 1, cex = 0.8
# ) } # Close the PDF device *after* the loop dev.off()

Now printing the fish that were fitted with higher order-polynomials in the htlm document

par(mfrow = c(1, 2), mar = c(4, 4, 4, 2))

pcrit_model_meta <- slope_tidy %>%
    dplyr::group_by(id) %>%
    dplyr::slice(1) %>%
    dplyr::ungroup() %>%
    dplyr::left_join(pcrit_model_df, ., by = "id")

higher_order_ids <- pcrit_model_meta %>%
    dplyr::filter(best_mod_type == "lm_2" | best_mod_type == "lm_3") %>%
    dplyr::distinct(id) %>%
    dplyr::pull()

for (id_i in higher_order_ids) {

    comment <- pcrit_model_meta %>%
        dplyr::filter(id == id_i) %>%
        dplyr::slice(1) %>%
        dplyr::mutate(comment = if_else(is.na(comments), "", paste0("#", comments))) %>%
        pull(comment)

    r2 <- pcrit_model_meta %>%
        dplyr::filter(id == id_i) %>%
        dplyr::mutate(pcrit_r2 = round(pcrit_r2, 3)) %>%
        dplyr::pull(pcrit_r2)

    conforming <- pcrit_model_meta %>%
        dplyr::filter(id == id_i) %>%
        dplyr::mutate(pcrit_nb_mo2_conforming = round(pcrit_nb_mo2_conforming, 3)) %>%
        dplyr::pull(pcrit_nb_mo2_conforming)

    P <- pcrit_model_meta %>%
        dplyr::filter(id == id_i) %>%
        dplyr::mutate(pcrit_p = round(pcrit_p, 3)) %>%
        dplyr::pull(pcrit_p)

    SMR <- pcrit_model_meta %>%
        dplyr::filter(id == id_i) %>%
        dplyr::mutate(pcrit_smr = round(pcrit_smr, 3)) %>%
        dplyr::pull(pcrit_smr)

    lowestMO2 <- pcrit_model_meta %>%
        dplyr::filter(id == id_i) %>%
        dplyr::mutate(pcrit_lowestMO2 = round(pcrit_lowestMO2, 3)) %>%
        dplyr::pull(pcrit_lowestMO2)

    # Generate and render the plot
    plotO2crit(o2critobj = pcrit_models[[id_i]])

    # Add a title
    mtext(text = paste0(id_i, " ", comment), side = 3, line = 2, adj = 0, col = "blue",
        font = 2, cex = 1.2)

    mtext(text = paste0("R2 = ", r2, "; p = ", P, "; CP < SMR = ", conforming), side = 3,
        line = 1, adj = 0, col = "blue", font = 0.5, cex = 0.8)

    mtext(text = paste0("SMR = ", SMR, "; lowestMO2 = ", lowestMO2), side = 3, line = 0,
        adj = 0, col = "blue", font = 0.5, cex = 0.8)
}

10.4.2 Visual inspection

We conducted visual inspection of all 58 fish to determine if O2crits were present. We have loaded this as a data frame 💿 o2crit_check

For this visual inspection, we grouped fish into “no”, “maybe” or “yes”, based on the certainty at which we observed a O2crits. In the case of yes or maybe, we also estimate dissolved oxygen percentage. All authors did this independently.

10.4.2.1 📈 Results

Let’s figure out how consistent the categorisation was, and what the majority response was. There were 35 cases with no disagreements (60.34%), 23 cases with one or more disagreements (39.66%).

disagreement_summary <- o2crit_check %>%
    group_by(id) %>%
    dplyr::reframe(n_disagreement = n_distinct(pcrit_cat) - 1, cat_disagreement = if_else(n_disagreement ==
        0, 0, 1), majority_cat = names(sort(table(pcrit_cat), decreasing = TRUE))[1],
        certainty = max(table(pcrit_cat)/5 * 100))

total_assesment <- o2crit_check %>%
    distinct(id) %>%
    nrow()

disagreement_summary %>%
    group_by(cat_disagreement) %>%
    dplyr::reframe(n = length(id), percent = round(n/total_assesment * 100, 2)) %>%
    gt()
cat_disagreement n percent
0 35 60.34
1 23 39.66

10.4.2.2 Figure S10

Here’s a heat map to show decisions for each fish

# Define custom colours for pcrit_cat
category_colours <- c(
  "no" = "#E03C32",
  "maybe" = "#FFD301",
  "yes" = "#7BB662"
)

# Create heatmap
ggplot(o2crit_check, aes(x = expert, y = id, fill = pcrit_cat)) +
  geom_tile(colour = "white") +
  scale_fill_manual(values = category_colours, na.value = "grey90") +
  labs(
    title = "Visual Assessment of pcrit_cat by Expert and ID",
    x = "Expert",
    y = "Fish ID",
    fill = "Category"
  ) +
  theme_minimal() +
  theme(
    axis.text.y = element_text(size = 7), # adjust size as needed
    axis.text.x = element_text(angle = 45, hjust = 1),
    panel.grid = element_blank()
  )

Expert deviation from the majority assessment ranged from 22.41 to 6.90%.

o2crit_check_sum <- o2crit_check %>%
    dplyr::left_join(., disagreement_summary, by = "id") %>%
    dplyr::mutate(deviated = if_else(pcrit_cat != majority_cat, 1, 0))

o2crit_check_sum %>%
    group_by(expert) %>%
    summarise(deviations = sum(deviated), deviation_rate = round(deviations/58 *
        100, 2)) %>%
    dplyr::arrange(desc(deviations)) %>%
    gt()
expert deviations deviation_rate
ech 13 22.41
jmm 8 13.79
llk 6 10.34
tdc 5 8.62
mg 4 6.90

Let’s make a list of those that had majority assigned as yes for an O2crit. There was 10 total.

o2crit_visual_yes <- disagreement_summary %>%
    dplyr::filter(majority_cat == "yes") %>%
    dplyr::pull(id)

paste0("Based on the majority visual assessment ", length(o2crit_visual_yes), " fish have possible O₂crit.")
## [1] "Based on the majority visual assessment 10 fish have possible O₂crit."

10.4.3 O₂crit numerical rules

We also used a numerical rule to asses if a O2crit was observed.

We filtered for only cases were at the lowest O₂ value three consecutive MO₂ measures full below our SMR and fifth percentile of the MO2 values observed at dissolved O2 levels > 80%. In the model output these are called nb_mo2_conforming points.

o2crit_numerical_yes <- pcrit_model_df %>%
    dplyr::filter(pcrit_nb_mo2_conforming > 2) %>%
    pull(id)

paste0("Based on this rule there are ", length(o2crit_numerical_yes), " fish with possible O₂crits.")
## [1] "Based on this rule there are 16 fish with possible O₂crits."

10.4.4 O₂crit numbers

We will now check to see which fish have O2crit based on the numerical and visual inspections.

There are eight fish that both numerically and visually were determined to have a O2crit. Two fish with a visually confirmed O2crit, that did not meet the numerical criteria. Eight fish meet the numerical criteria, but did not pass the visual inspection.

shared_elements <- intersect(o2crit_visual_yes, o2crit_numerical_yes)

unique_to_visual <- setdiff(o2crit_visual_yes, o2crit_numerical_yes)

unique_to_numerical <- setdiff(o2crit_numerical_yes, o2crit_visual_yes)

list(Shared = shared_elements, Unique_to_visual = unique_to_visual, Unique_to_numerical = unique_to_numerical)
## $Shared
## [1] "a_0_26nov_4" "b_0_24nov_1" "b_0_25nov_1" "b_0_25nov_3" "b_0_26nov_1"
## [6] "b_9_21nov_1" "b_9_21nov_2" "b_9_21nov_3"
## 
## $Unique_to_visual
## [1] "c_0_21nov_4" "d_0_21nov_2"
## 
## $Unique_to_numerical
## [1] "a_9_21nov_3" "b_0_24nov_2" "b_0_26nov_3" "b_9_22nov_1" "c_0_21nov_2"
## [6] "c_9_24nov_2" "d_0_21nov_3" "d_9_25nov_3"

10.4.4.1 📈 Results

Looking at mean O2crit estimates.

pop_mass <- pcrit_model_meta %>%
    dplyr::reframe(mean_mass = mean(mass)) %>%
    pull(mean_mass)

pcrit_model_meta %>%
    dplyr::filter(id %in% shared_elements) %>%
    dplyr::mutate(rel_mass = mass/pop_mass) %>%
    dplyr::reframe(n_o2crit = length(pcrit_vaule), mean_o2crit = mean(pcrit_vaule),
        min_o2crit = min(pcrit_vaule), max_o2crit = max(pcrit_vaule), mean_mass = mean(mass),
        mean_rel_mass = mean(rel_mass), sd_mass = sd(mass), sd_rel_mass = sd(rel_mass)) %>%
    gt() %>%
    fmt_number(columns = where(is.numeric), decimals = 2)
n_o2crit mean_o2crit min_o2crit max_o2crit mean_mass mean_rel_mass sd_mass sd_rel_mass
8.00 24.94 19.10 32.00 0.46 0.85 0.18 0.34
min_dos <- slope_tidy %>%
    dplyr::filter(!(id %in% shared_elements)) %>%
    group_by(id) %>%
    dplyr::reframe(min_do = min(DO))

min_dos %>%
    dplyr::reframe(mean_min_do = round(mean(min_do), 2), sd_min_do = round(sd(min_do),
        2), min_min_do = round(min(min_do), 2)) %>%
    gt()
mean_min_do sd_min_do min_min_do
13.12 3.23 8.82
pcrit_model_meta %>%
    dplyr::filter(id %in% shared_elements) %>%
    dplyr::group_by(salinity_group) %>%
    dplyr::reframe(n_o2crit = length(pcrit_vaule), mean_o2crit = mean(pcrit_vaule),
        min_o2crit = min(pcrit_vaule), max_o2crit = max(pcrit_vaule), mean_mass = mean(mass)) %>%
    gt()
salinity_group n_o2crit mean_o2crit min_o2crit max_o2crit mean_mass
0 5 25.32 19.1 32.0 0.3540000
9 3 24.30 21.7 28.9 0.6466667

These are the 8 fish that meet numerical criteria and passed visual inspection.

par(mfrow = c(1, 2), mar = c(4, 4, 4, 2))

for (id_i in shared_elements) {

    comment <- pcrit_model_meta %>%
        dplyr::filter(id == id_i) %>%
        dplyr::slice(1) %>%
        dplyr::mutate(comment = if_else(is.na(comments), "", paste0("#", comments))) %>%
        pull(comment)

    r2 <- pcrit_model_meta %>%
        dplyr::filter(id == id_i) %>%
        dplyr::mutate(pcrit_r2 = round(pcrit_r2, 3)) %>%
        dplyr::pull(pcrit_r2)

    conforming <- pcrit_model_meta %>%
        dplyr::filter(id == id_i) %>%
        dplyr::mutate(pcrit_nb_mo2_conforming = round(pcrit_nb_mo2_conforming, 3)) %>%
        dplyr::pull(pcrit_nb_mo2_conforming)

    P <- pcrit_model_meta %>%
        dplyr::filter(id == id_i) %>%
        dplyr::mutate(pcrit_p = round(pcrit_p, 3)) %>%
        dplyr::pull(pcrit_p)

    SMR <- pcrit_model_meta %>%
        dplyr::filter(id == id_i) %>%
        dplyr::mutate(pcrit_smr = round(pcrit_smr, 3)) %>%
        dplyr::pull(pcrit_smr)

    lowestMO2 <- pcrit_model_meta %>%
        dplyr::filter(id == id_i) %>%
        dplyr::mutate(pcrit_lowestMO2 = round(pcrit_lowestMO2, 3)) %>%
        dplyr::pull(pcrit_lowestMO2)

    # Generate and render the plot
    plotO2crit(o2critobj = pcrit_models[[id_i]])

    # Add a title
    mtext(text = paste0(id_i, " ", comment), side = 3, line = 2, adj = 0, col = "blue",
        font = 2, cex = 1.2)

    mtext(text = paste0("R2 = ", r2, "; p = ", P, "; CP < SMR = ", conforming, "; SMR = ",
        SMR, "; lowestMO2 = ", lowestMO2), side = 3, line = 1, adj = 0, col = "blue",
        font = 1, cex = 0.8)
}

Pulling model method and lethal points

ids <- pcrit_model_meta %>%
    dplyr::distinct(id) %>%
    dplyr::pull()

pcrit_mean_plot <- list()

for (id_i in ids) {
    best_mod_type <- best_mod %>%
        dplyr::filter(id == id_i) %>%
        pull(model_type)
    mod <- pcrit_models[[id_i]]
    data <- mod$origData
    min_leathal <- min(mod$lethalPoints)
    method <- mod$Method
    pcrit <- mod$o2crit
    data_slope <- data %>%
        dplyr::mutate(n_row = 1:nrow(.), leathal = if_else(n_row >= min_leathal,
            "nonregulation", "regulation"), method = method, pcrit = pcrit, best_mod_type = best_mod_type)
    pcrit_mean_plot[[id_i]] <- data_slope
}

pcrit_mean_plot_df <- bind_rows(pcrit_mean_plot)

10.4.4.2 Figure S11

Making a O2crit plot for each of the eight fish that past visual and numerical assessment on the same axis.

pcrit_summary <- pcrit_mean_plot_df %>%
    dplyr::filter(id %in% shared_elements) %>%
    group_by(id) %>%
    summarise(SMR = first(SMR), pcrit = first(pcrit), max_DO = max(DO, na.rm = TRUE),
        method = first(method), .groups = "drop")

smooth_segments <- map_dfr(unique(pcrit_summary$id), function(cur_id) {

    sum_info <- pcrit_summary %>%
        filter(id == cur_id)
    SMR_val <- sum_info$SMR
    method_val <- sum_info$method

    df <- pcrit_mean_plot_df %>%
        filter(id == cur_id, leathal == "nonregulation")

    if (nrow(df) < 2)
        return(NULL)

    # Fit the appropriate model
    if (method_val == "through_origin") {
        model <- lm(MO2 ~ DO + 0, data = df)
    } else {
        model <- lm(MO2 ~ DO, data = df)
    }

    coefs <- coef(model)

    if (method_val == "through_origin") {
        slope <- coefs[1]
        DO_start <- 0
        DO_stop <- SMR_val/slope
    } else {
        intercept <- coefs[1]
        slope <- coefs[2]
        # For LS_reg, start at 0 if prediction is already positive, or at the
        # DO where predicted MO2 becomes 0 if not.
        DO_start <- if (intercept < 0)
            -intercept/slope else 0
        DO_stop <- (SMR_val - intercept)/slope
    }

    # IMPORTANT: Do not bound DO_stop by max(df$DO) if you want to extrapolate
    # DO_stop <- min(DO_stop, max(df$DO, na.rm = TRUE)) # <-- Remove or comment
    # out this line

    # Create a sequence from DO_start to DO_stop
    DO_seq <- seq(DO_start, DO_stop, length.out = 100)

    # Calculate the predicted MO2 values for this sequence
    MO2_pred <- if (method_val == "through_origin") {
        slope * DO_seq
    } else {
        intercept + slope * DO_seq
    }

    tibble(id = cur_id, DO = DO_seq, MO2_pred = MO2_pred)
})

Fig_4 <- ggplot() + geom_point(data = pcrit_mean_plot_df %>%
    dplyr::filter(id %in% shared_elements), aes(x = DO, y = MO2, colour = id, alpha = leathal)) +
    scale_alpha_manual(values = c(regulation = 0.2, nonregulation = 0.8)) + guides(alpha = "none") +
    geom_line(data = smooth_segments, aes(x = DO, y = MO2_pred, colour = id), size = 1.2) +
    geom_segment(data = pcrit_summary, aes(x = pcrit, xend = max_DO, y = SMR, yend = SMR)) +
    geom_segment(data = pcrit_summary, aes(x = pcrit, xend = pcrit, y = 0, yend = SMR)) +
    scale_colour_viridis_d(option = "plasma") + theme_classic() + labs(y = expression("mg O"[2] ~
    "h"^-1), x = "Dissolved oxygen (DO; % saturation)") + theme(legend.position = "bottom")

Fig_4

Figure S10: Oxygen consumption rate (ṀO₂; mg⁻¹ O₂ h⁻¹) and ambient dissolved oxygen (DO; % saturation) plotted for each fish that passed both past visual and numerical assessment for O2crit. The horizontal dashed lines show SMR, the solid sloped lines represent the linear best fit for ṀO values below regulatory failure. The vertical soild lines represent O2crit (the intersection of the linear best fit for ṀO values below regulatory failure and SMR).

Let’s make some example plots for each major types of O2-MO2 relationships (i.e. 0th-, 1st-, and 2nd/3rd- order polynomial).

First an example of a 0th-order polynomial fit (c_9_24nov_4).

fig_2a <- bayes_reg_mods_predictions %>%
  dplyr::filter(id == "c_9_24nov_4", model_type == "lm_0") %>% 
  ggplot() +
  geom_point(aes(x = DO, y = MO2), colour = "black", size = 3, alpha = 0.5) +
  geom_segment(
    aes(x = 0, xend = 100, y = 0, yend = SMR[1]), # SMR
    linetype = "dashed", size = 1, colour = "#0F7AAF") +
  geom_segment(
    aes(x = 0, xend = 100, y = SMR[1], yend = SMR[1]), # conform
    linetype = "dashed", size = 1, colour = "#FFA500") +
  theme_few() +
    labs(
    y = expression("mg O"[2]~"h"^-1),
    x = NULL
  ) +
  theme(
    axis.title.x = element_blank(),
    axis.text.x = element_blank(),
    axis.ticks.x = element_blank()
  )

Second an example of a 1st-order polynomial best fit (d_9_25nov_3).

fig_2b <- bayes_reg_mods_predictions %>%
  dplyr::filter(id == "d_9_25nov_3", model_type == "lm_1") %>% 
  ggplot() +
  geom_point(aes(x = DO, y = MO2), colour = "black", size = 3, alpha = 0.5) +
  # geom_line(aes(x = DO, y = predicted * mass), size = 2, colour = "blue") +
  geom_segment(
    aes(x = 0, xend = 100, y = 0, yend = SMR[1]), # SMR
    linetype = "dashed", size = 1, colour = "#0F7AAF") +
  geom_segment(
    aes(x = 0, xend = 100, y = SMR[1], yend = SMR[1]), # conform
    linetype = "dashed", size = 1, colour = "#FFA500") +
  theme_few() +
  labs(
    y = expression("mg O"[2]~"h"^-1),
    x = NULL
  ) +
  theme(
    axis.text.x = element_blank(),
    axis.ticks.x = element_blank()
  )

Last an example of a fish best fit with a higher order polynomial (2nd-order).

# Define the example ID
pcrit_example <- "b_0_24nov_1"

# 1. Prepare summary data for the selected ID
pcrit_summary_example <- pcrit_mean_plot_df %>%
  filter(id == pcrit_example) %>%
  summarise(
    SMR = first(SMR),
    pcrit = first(pcrit),
    max_DO = max(DO, na.rm = TRUE),
    method = first(method),
    .groups = "drop"
  )

# 2. Subset the original nonregulation data
df <- pcrit_mean_plot_df %>%
  filter(id == pcrit_example, leathal == "nonregulation")

# 3. Fit model and generate predictions
if (nrow(df) >= 2) {
  SMR_val <- pcrit_summary_example$SMR
  method_val <- pcrit_summary_example$method
  
  if (method_val == "through_origin") {
    model <- lm(MO2 ~ DO + 0, data = df)
    slope <- coef(model)[1]
    DO_start <- 0
    DO_stop <- SMR_val / slope
    MO2_pred <- slope * seq(DO_start, DO_stop, length.out = 100)
  } else {
    model <- lm(MO2 ~ DO, data = df)
    intercept <- coef(model)[1]
    slope <- coef(model)[2]
    DO_start <- if (intercept < 0) -intercept / slope else 0
    DO_stop <- (SMR_val - intercept) / slope
    DO_seq <- seq(DO_start, DO_stop, length.out = 100)
    MO2_pred <- intercept + slope * DO_seq
  }
  
  DO_seq <- seq(DO_start, DO_stop, length.out = 100)
  
  smooth_segment <- tibble(
    id = pcrit_example,
    DO = DO_seq,
    MO2_pred = MO2_pred
  )
}

# 4. Plot the figure
fig_2c <- ggplot() +
  geom_point(
    data = pcrit_mean_plot_df %>% filter(id == pcrit_example),
    aes(x = DO, y = MO2, colour = leathal), size = 3, alpha = 0.5
  ) +
  scale_colour_manual(values = c("regulation" = "black", "nonregulation" = "#c30010")) +
  
  # geom_smooth(data = pcrit_mean_plot_df %>% filter(id == pcrit_example),
  #             aes(x = DO, y = MO2), method = "lm", formula = y ~ poly(x, 3), se = FALSE, alpha = 0.8, size = 2, fullrange = TRUE) +
  
  geom_line(
    data = smooth_segment,
    aes(x = DO, y = MO2_pred), size = 2, colour = "#c30010" #failure  slope
  ) +
  
  
  geom_segment(
    data = pcrit_summary_example,
    aes(x = 0, xend = max_DO, y = SMR, yend = SMR), #SMR
    colour = "#FFA500", linetype = "dashed", size = 1) +
  
  geom_segment(
    data = pcrit_summary_example,
    aes(x = 0, xend = max_DO, y = 0, yend = SMR), #confrom
    linetype = "dashed", size = 1, colour = "#0F7AAF") +
  
  geom_segment(
    data = pcrit_summary_example,
    aes(x = pcrit, xend = pcrit, y = 0, yend = SMR), colour = "darkgreen", size = 2 #Pcrit
  ) +
  
  #coord_cartesian(ylim = c(0, 0.16)) +
  theme_few() +
   labs(
    y = expression("mg O"[2]~"h"^-1),
    x = "Dissolved oxygen (DO; % saturation)"
  ) +
  theme(legend.position = "none")

10.4.4.3 Figure S12 (Figure 2)

Here we are showing the out put of the Chabot O2crit model,

fig_2_bind <- grid.arrange(fig_2a, fig_2b, fig_2c, ncol = 1)  #bind plot

Figure 2: Figure 2. Representative examples of fish that were best modelled by a 0th-order polynomial (A), 1st-order polynomial (B), and 2nd-order polynomial (C). The orange dashed line shows the SMR, while the blue dashed line shows the theoretical oxyconfroming regression. For plot (C), the red sloped line shows the O2 regulation failure regression (based on the rule-based linear regression method of Claireaux and Chabot, 2016), and the solid horizontal green line shows the O2crit estimate (intersection of O2 failure regression and SMR).

# ggsave(filename = paste0(output_fig_wd, './ms_figures./figure-2.pdf'), plot =
# fig_2_bind, width = 210, height = 297, units = 'mm')

11 Comapring to past data

Here, we have recreated a similar plot to that presented in Urbina, Glover, and Forster (2012)[1] and have extracted the mean level data from Figure 1a using the metaDigitise package in R[3]. This data is called urbina_et_al_2012. This allows us to compare the differences in Mo2 and the relationship between Mo2 and O2.

First making a binned data frame to match Urbina, Glover, and Forster (2012) as closely as possible.

min_o2_kpa <- min(slope_tidy$o2_kpa, na.rm = TRUE)
max_o2_kpa <- max(slope_tidy$o2_kpa, na.rm = TRUE)

time_bin_df <- slope_tidy %>%
  mutate(o2_group = cut(o2_kpa, 
                        breaks = seq(min_o2_kpa, max_o2_kpa, length.out = 13), # 11 intervals, so 12 breakpoints
                        labels = paste0("Group ", 1:12), 
                        include.lowest = TRUE)) %>% 
  dplyr::group_by(o2_group) %>% 
  dplyr::reframe(mean_MO2_g = mean(MO2_g)*31.25,
                 mean_o2_kpa = mean(o2_kpa),
                 n = length(MO2_g)*31.25,
                 MO2_g_sd = sd(MO2_g)*31.25,
                 o2_kpa_sd = sd(o2_kpa))

11.1 Figure S13 (Figure 3)

Now the plot with our mean data and the mean data from Urbina, Glover, and Forster (2012)[1]

n <- slope_tidy %>%
    dplyr::distinct(id) %>%
    nrow(.)

# Existing plot
fig3 <- time_bin_df %>%
    ggplot() + geom_point(data = slope_tidy, aes(y = MO2_g * 31.25, x = o2_kpa),
    size = 2, color = "grey", alpha = 0.3) + geom_point(data = time_bin_df, aes(y = mean_MO2_g,
    x = mean_o2_kpa), size = 3, colour = "#0E4C92", show.legend = FALSE) + geom_errorbar(data = time_bin_df,
    aes(y = mean_MO2_g, x = mean_o2_kpa, ymin = mean_MO2_g - MO2_g_sd, ymax = mean_MO2_g +
        MO2_g_sd), width = 0.15, colour = "#0E4C92") + geom_errorbarh(data = time_bin_df,
    aes(y = mean_MO2_g, x = mean_o2_kpa, xmin = mean_o2_kpa - o2_kpa_sd, xmax = mean_o2_kpa +
        o2_kpa_sd), height = 0.4, colour = "#0E4C92") + geom_point(data = urbina_et_al_2012,
    aes(x = o2_mean, y = mo2_mean), size = 3, shape = 1, fill = "#D21F3C", color = "#D21F3C",
    stroke = 1) + geom_errorbar(data = urbina_et_al_2012, aes(x = o2_mean, ymin = mo2_mean -
    mo2_sd, ymax = mo2_mean + mo2_sd), width = 0.2, colour = "#D21F3C") + geom_errorbarh(data = urbina_et_al_2012,
    aes(y = mo2_mean, xmin = o2_mean - o2_sd, xmax = o2_mean + o2_sd), height = 0.4,
    colour = "#D21F3C") + annotate("text", x = 0, y = 16, label = bquote(atop("Current data (blue), " *
    italic(n) * " = " * .(n), "Urbina data (red), " * italic(n) * " = 67")), hjust = 0,
    vjust = 1, size = 4) + theme(legend.position = "none", panel.grid.major = element_blank(),
    panel.grid.minor = element_blank(), panel.border = element_rect(colour = "black",
        fill = NA, linewidth = 1), panel.background = element_rect(fill = "white",
        colour = NA), plot.background = element_rect(fill = "white", colour = NA)) +
    labs(subtitle = NULL, x = expression("PO"[2] ~ "(kPa)"), y = expression("MO"[2] ~
        "(μmol O"[2] ~ "g"^-1 ~ "h"^-1)) + scale_y_continuous(limits = c(0, 16),
    breaks = seq(0, 16, by = 2)) + scale_x_continuous(limits = c(0, 22), breaks = seq(0,
    22, by = 2))

fig3

Figure 3: Mean and standard error of metabolic rate (MO2 μmol O2 g-1 h-1) and oxygen concentration (PO2 kPa) using 12 evenly spaced bins over the O2 range of observed data (blue filled dots). Compared against the mean and standard error reported in Urbina (2012)[1] (red open dots). The grey dots are the raw observed data form the present study.

# ggsave(filename = paste0(output_fig_wd, './ms_figures./figure-3.pdf'), plot =
# fig3, width = 210, height = 297/2, # Specify the height of the plot units =
# 'mm')

12 📚 References

[1] Claireaux, G. and Chabot, D. (2016) Responses by fishes to environmental hypoxia: integration through Fry’s concept of aerobic metabolic scope. Journal of Fish Biology https://doi.org/10.1111/jfb.12833

[2] Urbina MA, Glover CN, and Forster ME, (2012) A novel oxyconforming response in the freshwater fish Galaxias maculatus. Comparative Biochemistry and Physiology Part A: Molecular & Integrative Physiology. https://doi.org/10.1016/j.cbpa.2011.11.011

[3] Pick JL, Nakagawa S, and Noble DWA (2018) Reproducible, flexible and high-throughput data extraction from primary literature: The metaDigitise r package. https://doi.org/10.1111/2041-210X.13118

13 💻 Session information

citation()
## 
## To cite R in publications use:
## 
##   R Core Team (2023). R: A language and environment for statistical
##   computing. R Foundation for Statistical Computing, Vienna, Austria.
##   URL https://www.R-project.org/.
## 
## A BibTeX entry for LaTeX users is
## 
##   @Manual{,
##     title = {R: A Language and Environment for Statistical Computing},
##     author = {{R Core Team}},
##     organization = {R Foundation for Statistical Computing},
##     address = {Vienna, Austria},
##     year = {2023},
##     url = {https://www.R-project.org/},
##   }
## 
## We have invested a lot of time and effort in creating R, please cite it
## when using it for data analysis. See also 'citation("pkgname")' for
## citing R packages.

Here is a detailed list of the session information

sessioninfo::session_info()
## ─ Session info ───────────────────────────────────────────────────────────────
##  setting  value
##  version  R version 4.2.3 (2023-03-15 ucrt)
##  os       Windows 10 x64 (build 19044)
##  system   x86_64, mingw32
##  ui       RTerm
##  language (EN)
##  collate  English_Australia.utf8
##  ctype    English_Australia.utf8
##  tz       Australia/Sydney
##  date     2025-04-02
##  pandoc   3.1.1 @ C:/Program Files/RStudio/resources/app/bin/quarto/bin/tools/ (via rmarkdown)
## 
## ─ Packages ───────────────────────────────────────────────────────────────────
##  package           * version  date (UTC) lib source
##  abind               1.4-5    2016-07-21 [1] CRAN (R 4.2.0)
##  arrayhelpers        1.1-0    2020-02-04 [1] CRAN (R 4.2.3)
##  backports           1.4.1    2021-12-13 [1] CRAN (R 4.2.0)
##  bayesplot         * 1.11.1   2024-02-15 [1] CRAN (R 4.2.3)
##  bitops              1.0-7    2021-04-24 [1] CRAN (R 4.2.0)
##  bridgesampling      1.1-2    2021-04-16 [1] CRAN (R 4.2.3)
##  brms              * 2.21.0   2024-03-20 [1] CRAN (R 4.2.3)
##  Brobdingnag         1.2-9    2022-10-19 [1] CRAN (R 4.2.3)
##  broom               1.0.4    2023-03-11 [1] CRAN (R 4.2.3)
##  broom.helpers       1.15.0   2024-04-05 [1] CRAN (R 4.2.3)
##  broom.mixed       * 0.2.9.5  2024-04-01 [1] CRAN (R 4.2.3)
##  bslib               0.7.0    2024-03-29 [1] CRAN (R 4.2.3)
##  cachem              1.0.7    2023-02-24 [1] CRAN (R 4.2.3)
##  caTools             1.18.2   2021-03-28 [1] CRAN (R 4.2.3)
##  cellranger          1.1.0    2016-07-27 [1] CRAN (R 4.2.3)
##  checkmate           2.3.1    2023-12-04 [1] CRAN (R 4.2.3)
##  cli                 3.6.4    2025-02-13 [1] CRAN (R 4.2.3)
##  coda                0.19-4.1 2024-01-31 [1] CRAN (R 4.2.3)
##  codetools           0.2-19   2023-02-01 [2] CRAN (R 4.2.3)
##  colorspace          2.1-0    2023-01-23 [1] CRAN (R 4.2.3)
##  crosstalk           1.2.1    2023-11-23 [1] CRAN (R 4.2.3)
##  curl                5.2.1    2024-03-01 [1] CRAN (R 4.2.3)
##  data.table        * 1.17.0   2025-02-22 [1] CRAN (R 4.2.3)
##  DEoptimR            1.1-3    2023-10-07 [1] CRAN (R 4.2.3)
##  devtools          * 2.4.5    2022-10-11 [1] CRAN (R 4.2.3)
##  digest              0.6.35   2024-03-11 [1] CRAN (R 4.2.3)
##  distributional      0.4.0    2024-02-07 [1] CRAN (R 4.2.3)
##  doParallel          1.0.17   2022-02-07 [1] CRAN (R 4.2.3)
##  dplyr             * 1.1.4    2023-11-17 [1] CRAN (R 4.2.3)
##  ellipsis            0.3.2    2021-04-29 [1] CRAN (R 4.2.3)
##  emmeans           * 1.11.0   2025-03-20 [1] CRAN (R 4.2.3)
##  estimability        1.5.1    2024-05-12 [1] CRAN (R 4.2.3)
##  evaluate            0.24.0   2024-06-10 [1] CRAN (R 4.2.3)
##  fansi               1.0.4    2023-01-22 [1] CRAN (R 4.2.3)
##  farver              2.1.1    2022-07-06 [1] CRAN (R 4.2.3)
##  fastmap             1.1.1    2023-02-24 [1] CRAN (R 4.2.3)
##  forcats           * 1.0.0    2023-01-29 [1] CRAN (R 4.2.3)
##  foreach             1.5.2    2022-02-02 [1] CRAN (R 4.2.3)
##  formatR             1.14     2023-01-17 [1] CRAN (R 4.2.3)
##  fs                  1.6.4    2024-04-25 [1] CRAN (R 4.2.3)
##  furrr               0.3.1    2022-08-15 [1] CRAN (R 4.2.3)
##  future            * 1.33.2   2024-03-26 [1] CRAN (R 4.2.3)
##  generics            0.1.3    2022-07-05 [1] CRAN (R 4.2.3)
##  ggdist              3.3.2    2024-03-05 [1] CRAN (R 4.2.3)
##  ggplot2           * 3.5.1    2024-04-23 [1] CRAN (R 4.2.3)
##  ggthemes          * 5.1.0    2024-02-10 [1] CRAN (R 4.2.3)
##  globals             0.16.3   2024-03-08 [1] CRAN (R 4.2.3)
##  glue                1.8.0    2024-09-30 [1] CRAN (R 4.2.3)
##  gridExtra         * 2.3      2017-09-09 [1] CRAN (R 4.2.3)
##  gsw                 1.2-0    2024-08-19 [1] CRAN (R 4.2.3)
##  gt                * 0.11.0   2024-07-09 [1] CRAN (R 4.2.3)
##  gtable              0.3.5    2024-04-22 [1] CRAN (R 4.2.3)
##  gtsummary         * 1.7.2    2023-07-15 [1] CRAN (R 4.2.3)
##  highr               0.11     2024-05-26 [1] CRAN (R 4.2.3)
##  hms               * 1.1.3    2023-03-21 [1] CRAN (R 4.2.3)
##  htmltools           0.5.8.1  2024-04-04 [1] CRAN (R 4.2.3)
##  htmlwidgets         1.6.4    2023-12-06 [1] CRAN (R 4.2.3)
##  httpuv              1.6.9    2023-02-14 [1] CRAN (R 4.2.3)
##  httr                1.4.7    2023-08-15 [1] CRAN (R 4.2.3)
##  inline              0.3.19   2021-05-31 [1] CRAN (R 4.2.3)
##  insight             1.1.0    2025-03-01 [1] CRAN (R 4.2.3)
##  iterators           1.0.14   2022-02-05 [1] CRAN (R 4.2.3)
##  janitor           * 2.2.0    2023-02-02 [1] CRAN (R 4.2.3)
##  jquerylib           0.1.4    2021-04-26 [1] CRAN (R 4.2.3)
##  jsonlite            1.8.8    2023-12-04 [1] CRAN (R 4.2.3)
##  knitr               1.42     2023-01-25 [1] CRAN (R 4.2.3)
##  labeling            0.4.3    2023-08-29 [1] CRAN (R 4.2.3)
##  later               1.3.0    2021-08-18 [1] CRAN (R 4.2.3)
##  lattice             0.20-45  2021-09-22 [2] CRAN (R 4.2.3)
##  lazyeval            0.2.2    2019-03-15 [1] CRAN (R 4.2.3)
##  lifecycle           1.0.4    2023-11-07 [1] CRAN (R 4.2.3)
##  listenv             0.9.1    2024-01-29 [1] CRAN (R 4.2.3)
##  loo                 2.8.0    2024-07-03 [1] CRAN (R 4.2.3)
##  lubridate         * 1.9.2    2023-02-10 [1] CRAN (R 4.2.3)
##  magrittr            2.0.3    2022-03-30 [1] CRAN (R 4.2.3)
##  marelac             2.1.11   2023-09-25 [1] CRAN (R 4.2.3)
##  marginaleffects   * 0.21.0   2024-06-14 [1] CRAN (R 4.2.3)
##  MASS                7.3-58.2 2023-01-23 [2] CRAN (R 4.2.3)
##  Matrix              1.5-3    2022-11-11 [2] CRAN (R 4.2.3)
##  matrixStats         1.3.0    2024-04-11 [1] CRAN (R 4.2.3)
##  mclust            * 6.1.1    2024-04-29 [1] CRAN (R 4.2.3)
##  measurements        1.5.1    2023-04-28 [1] CRAN (R 4.2.3)
##  memoise             2.0.1    2021-11-26 [1] CRAN (R 4.2.3)
##  mgcv                1.9-1    2023-12-21 [1] CRAN (R 4.2.3)
##  mime                0.12     2021-09-28 [1] CRAN (R 4.2.0)
##  miniUI              0.1.1.1  2018-05-18 [1] CRAN (R 4.2.3)
##  multcomp            1.4-25   2023-06-20 [1] CRAN (R 4.2.3)
##  munsell             0.5.1    2024-04-01 [1] CRAN (R 4.2.3)
##  mvtnorm             1.2-5    2024-05-21 [1] CRAN (R 4.2.3)
##  nlme                3.1-162  2023-01-31 [2] CRAN (R 4.2.3)
##  oce                 1.8-3    2024-08-17 [1] CRAN (R 4.2.3)
##  opdisDownsampling   1.0.1    2024-04-15 [1] CRAN (R 4.2.3)
##  pacman              0.5.1    2019-03-11 [1] CRAN (R 4.2.3)
##  parallelly          1.37.1   2024-02-29 [1] CRAN (R 4.2.3)
##  pbmcapply           1.5.1    2022-04-28 [1] CRAN (R 4.2.0)
##  performance       * 0.12.0   2024-06-08 [1] CRAN (R 4.2.3)
##  pillar              1.9.0    2023-03-22 [1] CRAN (R 4.2.3)
##  pkgbuild            1.4.4    2024-03-17 [1] CRAN (R 4.2.3)
##  pkgconfig           2.0.3    2019-09-22 [1] CRAN (R 4.2.3)
##  pkgload             1.3.4    2024-01-16 [1] CRAN (R 4.2.3)
##  plotly            * 4.10.4   2024-01-13 [1] CRAN (R 4.2.3)
##  plyr                1.8.8    2022-11-11 [1] CRAN (R 4.2.3)
##  posterior           1.6.0    2024-07-03 [1] CRAN (R 4.2.3)
##  pracma              2.4.4    2023-11-10 [1] CRAN (R 4.2.3)
##  profvis             0.3.8    2023-05-02 [1] CRAN (R 4.2.3)
##  promises            1.2.0.1  2021-02-11 [1] CRAN (R 4.2.3)
##  purrr             * 1.0.1    2023-01-10 [1] CRAN (R 4.2.3)
##  qqconf              1.3.2    2023-04-14 [1] CRAN (R 4.2.3)
##  qqplotr           * 0.0.6    2023-01-25 [1] CRAN (R 4.2.3)
##  QuickJSR            1.3.0    2024-07-08 [1] CRAN (R 4.2.3)
##  R6                  2.5.1    2021-08-19 [1] CRAN (R 4.2.3)
##  ragg                1.2.5    2023-01-12 [1] CRAN (R 4.2.3)
##  RColorBrewer        1.1-3    2022-04-03 [1] CRAN (R 4.2.0)
##  Rcpp              * 1.0.10   2023-01-22 [1] CRAN (R 4.2.3)
##  RcppParallel        5.1.8    2024-07-06 [1] CRAN (R 4.2.3)
##  readr             * 2.1.4    2023-02-10 [1] CRAN (R 4.2.3)
##  readxl            * 1.4.3    2023-07-06 [1] CRAN (R 4.2.3)
##  remotes             2.5.0    2024-03-17 [1] CRAN (R 4.2.3)
##  reshape2            1.4.4    2020-04-09 [1] CRAN (R 4.2.3)
##  respirometry      * 2.0.0    2024-07-18 [1] CRAN (R 4.2.3)
##  rlang               1.1.5    2025-01-17 [1] CRAN (R 4.2.3)
##  rmarkdown           2.21     2023-03-26 [1] CRAN (R 4.2.3)
##  robustbase          0.99-4-1 2024-09-27 [1] CRAN (R 4.2.3)
##  rstan             * 2.32.6   2024-03-05 [1] CRAN (R 4.2.3)
##  rstantools          2.4.0    2024-01-31 [1] CRAN (R 4.2.3)
##  rstudioapi          0.14     2022-08-22 [1] CRAN (R 4.2.3)
##  sandwich            3.1-0    2023-12-11 [1] CRAN (R 4.2.3)
##  sass                0.4.9    2024-03-15 [1] CRAN (R 4.2.3)
##  scales              1.3.0    2023-11-28 [1] CRAN (R 4.2.3)
##  seacarb             3.3.3    2024-02-15 [1] CRAN (R 4.2.3)
##  sessioninfo         1.2.2    2021-12-06 [1] CRAN (R 4.2.3)
##  shape               1.4.6.1  2024-02-23 [1] CRAN (R 4.2.3)
##  shiny               1.8.1.1  2024-04-02 [1] CRAN (R 4.2.3)
##  snakecase           0.11.1   2023-08-27 [1] CRAN (R 4.2.3)
##  SolveSAPHE          2.1.0    2021-05-01 [1] CRAN (R 4.2.0)
##  StanHeaders       * 2.32.9   2024-05-29 [1] CRAN (R 4.2.3)
##  stringi             1.7.12   2023-01-11 [1] CRAN (R 4.2.2)
##  stringr           * 1.5.1    2023-11-14 [1] CRAN (R 4.2.3)
##  survival            3.5-3    2023-02-12 [2] CRAN (R 4.2.3)
##  svUnit              1.0.6    2021-04-19 [1] CRAN (R 4.2.3)
##  systemfonts         1.0.4    2022-02-11 [1] CRAN (R 4.2.3)
##  tensorA             0.36.2.1 2023-12-13 [1] CRAN (R 4.2.3)
##  textshaping         0.3.6    2021-10-13 [1] CRAN (R 4.2.3)
##  TH.data             1.1-2    2023-04-17 [1] CRAN (R 4.2.3)
##  tibble            * 3.2.1    2023-03-20 [1] CRAN (R 4.2.3)
##  tidybayes         * 3.0.6    2023-08-12 [1] CRAN (R 4.2.3)
##  tidyr             * 1.3.0    2023-01-24 [1] CRAN (R 4.2.3)
##  tidyselect          1.2.1    2024-03-11 [1] CRAN (R 4.2.3)
##  tidyverse         * 2.0.0    2023-02-22 [1] CRAN (R 4.2.3)
##  timechange          0.2.0    2023-01-11 [1] CRAN (R 4.2.3)
##  twosamples          2.0.1    2023-06-23 [1] CRAN (R 4.2.3)
##  tzdb                0.3.0    2022-03-28 [1] CRAN (R 4.2.3)
##  urlchecker          1.0.1    2021-11-30 [1] CRAN (R 4.2.3)
##  usethis           * 2.2.3    2024-02-19 [1] CRAN (R 4.2.3)
##  utf8                1.2.3    2023-01-31 [1] CRAN (R 4.2.3)
##  V8                  4.4.2    2024-02-15 [1] CRAN (R 4.2.3)
##  vctrs               0.6.5    2023-12-01 [1] CRAN (R 4.2.3)
##  viridisLite         0.4.2    2023-05-02 [1] CRAN (R 4.2.3)
##  withr               3.0.0    2024-01-16 [1] CRAN (R 4.2.3)
##  xfun                0.38     2023-03-24 [1] CRAN (R 4.2.3)
##  xml2                1.3.6    2023-12-04 [1] CRAN (R 4.2.3)
##  xtable              1.8-4    2019-04-21 [1] CRAN (R 4.2.3)
##  yaml                2.3.7    2023-01-23 [1] CRAN (R 4.2.3)
##  zoo                 1.8-12   2023-04-13 [1] CRAN (R 4.2.3)
## 
##  [1] C:/R
##  [2] C:/Program Files/R/R-4.2.3/library
## 
## ──────────────────────────────────────────────────────────────────────────────
LS0tDQp0aXRsZTogIlRoZSByb2xlIG9mIG9zbW9yZXNwaXJhdG9yeSBjb21wcm9taXNlIGluIGh5cG94aWEgdG9sZXJhbmNlIG9mIHRoZSBwdXJwb3J0ZWRseSBveHljb25mb3JtaW5nIHRlbGVvc3QgR2FsYXhpYXMgbWFjdWxhdHVzIg0KYXV0aG9yOiAiSmFrZSBNYXJ0aW4iDQpkYXRlOiAiYHIgZm9ybWF0KFN5cy50aW1lKCksICclZCAlQiAlWScpYCINCm91dHB1dDoNCiAgaHRtbF9kb2N1bWVudDoNCiAgICBjb2RlX2Rvd25sb2FkOiB0cnVlDQogICAgY29kZV9mb2xkaW5nOiBoaWRlDQogICAgZGVwdGg6IDQNCiAgICBudW1iZXJfc2VjdGlvbnM6IHllcw0KICAgIHRoZW1lOiAgY29zbW8NCiAgICB0b2M6IHllcw0KICAgIHRvY19mbG9hdDogeWVzDQogICAgdG9jX2RlcHRoOiA0DQogIHBkZl9kb2N1bWVudDoNCiAgICB0b2M6IHllcw0Ka25pdDogfA0KICAoZnVuY3Rpb24oaW5wdXQsIC4uLikgew0KICAgIHJtYXJrZG93bjo6cmVuZGVyKA0KICAgICAgaW5wdXQsDQogICAgICBvdXRwdXRfZmlsZSA9IHBhc3RlMCgNCiAgICAgICAnaW5kZXguaHRtbCcNCiAgICAgICksDQogICAgICBlbnZpciA9IGdsb2JhbGVudigpDQogICAgKQ0KICB9KQ0KLS0tDQoNClshW0xpY2Vuc2U6IENDIEJZIDQuMF0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9iYWRnZS9MaWNlbnNlLUNDJTIwQlklMjA0LjAtbGlnaHRncmV5LnN2ZyldKGh0dHBzOi8vY3JlYXRpdmVjb21tb25zLm9yZy9saWNlbnNlcy9ieS80LjAvKQ0KDQo8IS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0+DQoNCiMg8J+TlSBSRUFEIE1FDQoNCjwhLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLT4NCg0KKirwn5qnIFVOUFVCTElTSEVEIEFORCBPTkdPSU5HIGV4YW1wbGUg8J+apyoqDQpUaGUgZGF0YSBpbiB0aGlzIHNjcmlwdCBhcmUgY3VycmVudGx5ICoqdW5wdWJsaXNoZWQqKiwgYW5kIHRoZSAqKmFuYWx5c2lzIGlzIG9uZ29pbmcqKi4gU29tZSBzZWN0aW9ucyBtYXkgYmUgaW5jb21wbGV0ZS4gVGhpcyBzY3JpcHQgaGFzIGJlZW4gbWFkZSBwdWJsaWNseSBhdmFpbGFibGUgdG8gZmFjaWxpdGF0ZSBzaGFyaW5nIHdpdGggY29sbGFib3JhdG9ycyBhbmQgY29sbGVhZ3VlcyB3aG8gYXJlIGludGVyZXN0ZWQgaW4gdGhlIHByb2dyZXNzIG9mIHRoZSBwcm9qZWN0IGFuZCB0byBlbnN1cmUgdHJhbnNwYXJlbmN5Lg0KDQoqKlNVTU1BUlkqKg0KVGhpcyBSIGNvZGUgZXN0aW1hdGVzIHRoZSByZWxhdGlvbnNoaXAgYmV0d2VlbiBveHlnZW4gY29uc3VtcHRpb24gKE1P4oKCKSBhbmQgYW1iaWVudCBveHlnZW4gcGFydGlhbCBwcmVzc3VyZSAoUE/igoIpIGluIHRoZSBDb21tb24gR2FsYXhpYXMgKCpHYWxheGlhcyBtYWN1bGF0dXMqKS4gSXQgYWxzbyBlc3RpbWF0ZXMgdGhlIGNyaXRpY2FsIHBhcnRpYWwgcHJlc3N1cmUgb2Ygb3h5Z2VuIGZvciBhZXJvYmljIG1ldGFib2xpc20gKFBjcml0KSwgY29tbW9ubHkgZGVmaW5lZCBhcyB0aGUgdGhyZXNob2xkIGJlbG93IHdoaWNoIHRoZSBveHlnZW4gY29uc3VtcHRpb24gcmF0ZSBjYW4gbm8gbG9uZ2VyIGJlIHN1c3RhaW5lZC4gVGhlIGFzc29jaWF0ZWQgYXJ0aWNsZSBpcyB0aXRsZWQgKiJUaGUgcm9sZSBvZiBvc21vcmVzcGlyYXRvcnkgY29tcHJvbWlzZSBpbiBoeXBveGlhIHRvbGVyYW5jZSBvZiB0aGUgcHVycG9ydGVkbHkgb3h5Y29uZm9ybWluZyB0ZWxlb3N0IEdhbGF4aWFzIG1hY3VsYXR1cy4iKiBJZiB5b3UgYXJlIHJlYWRpbmcgdGhlIEhUTUwgdmVyc2lvbiBvZiB0aGlzIHNjcmlwdCwgY2xpY2sgdGhlIGBDb2RlYCBidXR0b24gaW4gdGhlIHRvcCByaWdodCB0byBkb3dubG9hZCB0aGUgYC5SbWRgIGZpbGUuDQoNCioqQUlNKioNClRoZSBhcnRpY2xlIGFpbXMgdG8gdGVzdCB3aGV0aGVyICpHYWxheGlhcyBtYWN1bGF0dXMqIGNhbiBtYWludGFpbiBveHlnZW4gY29uc3VtcHRpb24gKE1P4oKCKSBhcyBhbWJpZW50IFBP4oKCIGRlY2xpbmVzIGFuZCwgaWYgc28sIGF0IHdoYXQgbGV2ZWwgaXQgcmVhY2hlcyB0aGUgY3JpdGljYWwgcGFydGlhbCBwcmVzc3VyZSBvZiBveHlnZW4gZm9yIGFlcm9iaWMgbWV0YWJvbGlzbSAoUGNyaXQpLg0KDQoqKkFVVEhPUlMqKg0KVGltb3RoeSBELiBDbGFyayBeW2FdXg0KTHVpcyBMLiBLdWNoZW5tw7xsbGVyIF5bYV1eDQpFbGl6YWJldGggQy4gSG9vdHMgXlthXV4NCk1hcnlhbmUgR3JhZGl0byBeW2FdXg0KSmFrZSBNLiBNYXJ0aW4gXlthLGJdXg0KKkluIG5vIHBhcnRpY3VsYXIgb3JkZXIqDQoNCioqQUZGSUxJQVRJT05TKioNClthXSBTY2hvb2wgb2YgTGlmZSBhbmQgRW52aXJvbm1lbnRhbCBTY2llbmNlcywgRGVha2luIFVuaXZlcnNpdHksIEdlZWxvbmcsIFZJQywgQXVzdHJhbGlhXA0KW2JdIFNjaG9vbCBvZiBCaW9sb2dpY2FsIFNjaWVuY2VzLCBNb25hc2ggVW5pdmVyc2l0eSwgQ2xheXRvbiwgVklDLCBBdXN0cmFsaWENCg0KKipDT05UUklCVVRPUiBST0xFUyoqDQrwn5qnICoqKlRvIGJlIGFkZGVkKioqIPCfmqcNCipCYXNlZCBvbiB0aGUgQ29udHJpYnV0b3IgUm9sZXMgVGF4b25vbXkgKENSZWRpVCkqDQoNCioqRElTQ0xBSU1FUioqDQpJIChKYWtlIE1hcnRpbikgYW0gZHlzbGV4aWMuIEkgaGF2ZSBtYWRlIGFuIGVmZm9ydCB0byByZXZpZXcgdGhlIHNjcmlwdCBmb3IgZ3JhbW1hdGljYWwgZXJyb3JzLCBidXQgc29tZSB3aWxsIGxpa2VseSByZW1haW4uIEkgYXBvbG9naXNlLiBQbGVhc2UgZmVlbCBmcmVlIHRvIHJlYWNoIG91dCB1c2luZyB0aGUgY29udGFjdCBkZXRhaWxzIGJlbG93IGlmIGFueXRoaW5nIGlzIHVuY2xlYXIuDQoNCjwhLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLT4NCg0KIyDwn5OnIENvbnRhY3QNCg0KPCEtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tPg0KDQoqKkpha2UgTS4gTWFydGluKioNCg0K8J+TpyAqKkVtYWlsOioqIFtqYWtlLm1hcnRpblxAZGVha2luLmVkdS5hdV0obWFpbHRvOmpha2UubWFydGluQGRlYWtpbi5lZHUuYXUpDQoNCvCfk6cgKipBbHQgRW1haWw6KiogW2pha2UubWFydGluLnJlc2VhcmNoXEBnbWFpbC5jb21dKG1haWx0bzpqYWtlLm1hcnRpbi5yZXNlYXJjaEBnbWFpbC5jb20pDQoNCvCfjJAgKipXZWI6KiogW2pha2VtYXJ0aW4ub3JnXShodHRwczovL2pha2VtYXJ0aW4ub3JnLykNCg0K8J+QmSAqKkdpdEh1YioqOiBbSmFrZU1hcnRpblJlc2VhcmNoXShodHRwczovL2dpdGh1Yi5jb20vSmFrZU1hcnRpblJlc2VhcmNoKQ0KDQo8IS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0+DQoNCiMg8J+TkSBTaGFyaW5nL2FjY2Vzc2luZyBhbmQgY2l0aW5nDQoNCjwhLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLT4NCg0KMS4gICoqTGljZW5zZXMvcmVzdHJpY3Rpb25zIHBsYWNlZCBvbiB0aGUgZGF0YToqKiBDQy1CWSA0LjANCg0KMi4gICoqTGluayB0byB0aGUgYXNzb2NpYXRlZCBwdWJsaWNhdGlvbjoqKlwNCiAgICDwn5qnICoqKlRvIGJlIGFkZGVkKioqIPCfmqcNCg0KMy4gICoqUmVjb21tZW5kZWQgY2l0YXRpb24gZm9yIHRoaXMgZGF0YToqKlwNCiAgICDwn5qnICoqKlRvIGJlIGFkZGVkKioqIPCfmqcNCg0KPCEtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tPg0KDQojIPCfk6YgUmVxdWlyZWQgcGFja2FnZXMgYW5kIGtuaXQgc2V0dGluZ3MNCg0KPCEtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tPg0KDQpBIGxpc3Qgb2YgdGhlIHJlcXVpcmVkIHBhY2thZ2VzIHRvIHJ1biB0aGUgc2NyaXB0Lg0KDQpgYGB7ciBzZXR1cH0NCiMgLS0tLSBJbnN0YWxsIHBhY21hbiBpZiBpdCdzIG5vdCBhbHJlYWR5IGluc3RhbGxlZCAtLS0tDQppZiAoIXJlcXVpcmVOYW1lc3BhY2UoInBhY21hbiIsIHF1aWV0bHkgPSBUUlVFKSkgaW5zdGFsbC5wYWNrYWdlcygicGFjbWFuIikNCg0KIyAtLS0tIExpc3Qgb2YgcmVxdWlyZWQgcGFja2FnZXMgLS0tLQ0KcGtncyA8LSBjKA0KICAjIC0tLS0tIERhdGEgVmlzdWFsaXNhdGlvbiAtLS0tLQ0KICAiZ2d0aGVtZXMiLCAiYmF5ZXNwbG90IiwgImd0IiwgImd0c3VtbWFyeSIsICJwbG90bHkiLCAicXFwbG90ciIsICJncmlkRXh0cmEiLA0KICANCiAgIyAtLS0tLSBUaWR5IERhdGEgYW5kIFdyYW5nbGluZyAtLS0tLQ0KICAidGlkeXZlcnNlIiwgImphbml0b3IiLCAicmVhZHhsIiwgImJyb29tLm1peGVkIiwgImRhdGEudGFibGUiLCAiaG1zIiwgImRldnRvb2xzIiwNCiAgIm1jbHVzdCIsDQogIA0KICAjIC0tLS0tIE1vZGVsbGluZyBhbmQgU3RhdGlzdGljYWwgQW5hbHlzaXMgLS0tLS0NCiAgImJybXMiLCAicnN0YW4iLCAibWFyZ2luYWxlZmZlY3RzIiwgInBlcmZvcm1hbmNlIiwgImVtbWVhbnMiLA0KICAidGlkeWJheWVzIiwgInJlc3Bpcm9tZXRyeSIsICJmdXR1cmUiDQopDQoNCiMgLS0tLSBJbnN0YWxsIGFuZCBsb2FkIGFsbCBwYWNrYWdlcyB1c2luZyBwYWNtYW4gLS0tLQ0Kc3VwcHJlc3NQYWNrYWdlU3RhcnR1cE1lc3NhZ2VzKA0KICBwYWNtYW46OnBfbG9hZChjaGFyID0gcGtncywgaW5zdGFsbCA9IFRSVUUpDQopDQoNCiNrbml0ZXIgc2VldHRpbmcNCmtuaXRyOjpvcHRzX2NodW5rJHNldCgNCm1lc3NhZ2UgPSBGQUxTRSwNCndhcm5pbmcgPSBGQUxTRSwgIyBubyB3YXJuaW5ncw0KY2FjaGUgPSBUUlVFLCMgQ2FjaGVpbmcgdG8gc2F2ZSB0aW1lIHdoZW4ga25pdGluZw0KdGlkeSA9IFRSVUUsDQpmaWcuYWxpZ24gPSAiY2VudGVyIiANCikNCmBgYA0KDQoNCjwhLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLT4NCg0KIyDwn5SnIEN1c3RvbSBmdW5jdGlvbnMNCg0KPCEtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tPg0KDQpIZXJlIGFyZSBzb21lIGN1c3RvbSBmdW5jdGlvbnMgdXNlZCB3aXRoaW4gdGhpcyBzY3JpcHQuDQoNCmBiYXllc19pbmNyZW1lbnRhbF9yZWdyZXNzaW9uX2J5X2lkKClgOiBBIGN1c3RvbSBmdW5jdGlvbiB0byBidWlsZCBCYXllc2lhbiBpbmNyZW1lbnRhbCByZWdyZXNzaW9ucy4gSXQgaXMgZGVzaWduZWQgdG8gcnVuIGEgbGlzdCBvZiBzdWJncm91cCBtb2RlbHMgKElEcykgaW4gcGFyYWxsZWwgdXNpbmcgNCBjb3Jlcy4gVGhlIGZ1bmN0aW9uIHVzZXMgYGJybSgpYCB3aXRoIGEgR2F1c3NpYW4gZXJyb3IgZGlzdHJpYnV0aW9uLg0KDQoqKlVzZSoqOlwNClRoZSBmdW5jdGlvbiBhY2NlcHRzIHRoZSBmb2xsb3dpbmcgYXJndW1lbnRzOlwNCi0gYGlkX2lgOiBBIGdyb3VwaW5nIGZhY3RvciBvciBJRCB1c2VkIHRvIGZpbHRlciB0aGUgZGF0YSBmb3IgZWFjaCByZWdyZXNzaW9uLiBJZiBub25lIGlzIHByb3ZpZGVkLCB0aGUgZnVuY3Rpb24gdXNlcyB0aGUgZW50aXJlIGRhdGFzZXQuXA0KLSBgaWRfbmFtZWA6IFRoZSBjb2x1bW4gbmFtZSAoYXMgYSBjaGFyYWN0ZXIgc3RyaW5nKSBjb3JyZXNwb25kaW5nIHRvIHRoZSBncm91cGluZyBmYWN0b3IgaW4gdGhlIGRhdGEgZnJhbWUuIElmIG5vdCBwcm92aWRlZCwgdGhlIGZ1bmN0aW9uIHVzZXMgYWxsIGRhdGEuXA0KLSBgZGF0YWA6IFRoZSBkYXRhIGZyYW1lIGNvbnRhaW5pbmcgdGhlIHZhcmlhYmxlcyBmb3IgYW5hbHlzaXMuXA0KLSBgcHJlZGljdG9yYDogVGhlIHByZWRpY3RvciB2YXJpYWJsZSBvZiBpbnRlcmVzdC5cDQotIGByZXNwb25zZWA6IFRoZSByZXNwb25zZSB2YXJpYWJsZSBvZiBpbnRlcmVzdC5cDQotIGBzZWVkX251bWJlcmA6IEEgcmFuZG9tIHNlZWQgdmFsdWUgZm9yIG1vZGVsIHJlcHJvZHVjaWJpbGl0eS5cDQotIGBzYXZlX21vZGVsc2A6IEEgbG9naWNhbCBhcmd1bWVudCBpbmRpY2F0aW5nIHdoZXRoZXIgdG8gc2F2ZSB0aGUgbW9kZWwgb3V0cHV0cyAoYFRSVUVgIG9yIGBGQUxTRWApLlwNCi0gYG1vZF9vdXRwdXRfd2RgOiBUaGUgb3V0cHV0IGRpcmVjdG9yeSB3aGVyZSBtb2RlbCBgLnJkc2AgZmlsZXMgc2hvdWxkIGJlIHNhdmVkICh1c2VkIG9ubHkgaWYgYHNhdmVfbW9kZWxzID0gVFJVRWApLg0KDQpgYGB7cn0NCiMgSW5zdGVhZCB3ZSBjb3VsZCBhbHNvIHVzZSBhIGRpc3RyaWJ1dGlvbmFsIHJlZ3Jlc3Npb24gYXBwcm9hY2gsIGJ5IHNwZWNpZmljYWxseSBtb2RlbGxpbmcgdGhlIHZhcmlhbmNlIGJ5IERPIChlLmcuIHNpZ21hIH4gRE8pLiBXZWlnaHRpbmcgbWF5IG5vdCBiZSByZXF1aXJlZCBpbiB0aGlzIGNhc2UsIEkgZG9uJ3QgdGhpbmsgaGlnaGVyIGRlbnNpdHkgb2YgdmFsdWVzIGluIGEgZ2l2ZW4gc3BhY2Ugd2lsbCBlZmZlY3QgQmF5ZXNpYW4gZXN0aW1hdGVzIGxpa2UgaXQgZG9lcyBpbiBmcmVxdWVudGlzdCBtb2RlbHMuIFNlZSBkaXNjb3Vyc2UgaHR0cHM6Ly9kaXNjb3Vyc2UubWMtc3Rhbi5vcmcvdC93ZWlnaHRzLWluLWJybS80Mjc4ICAgDQoNCmJheWVzX2luY3JlbWVudGFsX3JlZ3Jlc3Npb25fYnlfaWQgPC0gZnVuY3Rpb24oaWRfaSwgaWRfbmFtZSwgZGF0YSwgcHJlZGljdG9yLCByZXNwb25zZSwgc2VlZF9udW1iZXIsIHNhdmVfbW9kZWxzLCBtb2Rfb3V0cHV0X3dkKSB7DQogICMgSW5pdGlhdGUgYW4gZW1wdHkgbGlzdCB0byBzdG9yZSBtb2RlbHMNCiAgbW9kZWxzIDwtIGxpc3QoKQ0KICANCiAgIyBDaGVjayBpZiBpZF9uYW1lIGlzIG1pc3NpbmcsIE5VTEwsIG9yIGJsYW5rLCBhbmQgYXNzaWduIE5BIGlmIHNvDQogIGlmIChtaXNzaW5nKGlkX25hbWUpIHx8IGlzLm51bGwoaWRfbmFtZSkgfHwgaWRfbmFtZSA9PSAiIikgew0KICAgIGlkX25hbWUgPC0gTkENCiAgfQ0KICANCiAgIyBDaGVjayBpZiBpZF9pIGlzIG1pc3NpbmcsIE5VTEwsIG9yIGJsYW5rLCBhbmQgYXNzaWduIE5BIGlmIHNvDQogIGlmIChtaXNzaW5nKGlkX2kpIHx8IGlzLm51bGwoaWRfaSkgfHwgaWRfaSA9PSAiIikgew0KICAgIGlkX25hbWUgPC0gTkENCiAgfQ0KICANCiAgIyBGaWx0ZXIgZGF0YSBmb3IgdGhlIGN1cnJlbnQgSUQgaWYgaWRfbmFtZSBpcyBnaXZlbiBhcyBhIGZhY3RvciBvciBjaGFyYWN0ZXIgYW5kIGlkX2kgaXMgZGVmaW5lZA0KICBkZl9pIDwtIGRhdGEgJT4lDQogICAgZHBseXI6OmZpbHRlcigNCiAgICAgIGlmICghaXMubmEoaWRfaSkgJiYgKGlzLmZhY3RvcihkYXRhW1tpZF9uYW1lXV0pIHx8IGlzLmNoYXJhY3RlcihkYXRhW1tpZF9uYW1lXV0pKSkgew0KICAgICAgICAhIXJsYW5nOjpzeW0oaWRfbmFtZSkgPT0gaWRfaQ0KICAgICAgfSBlbHNlIHsNCiAgICAgICAgVFJVRQ0KICAgICAgfQ0KICAgICkNCiAgDQogICMgRHluYW1pY2FsbHkgY3JlYXRlIGZvcm11bGFzDQogIGZvcm11bGFfbG1fMCA8LSByZWZvcm11bGF0ZSgiMSIsIHJlc3BvbnNlKQ0KICBmb3JtdWxhX2xtXzEgPC0gcmVmb3JtdWxhdGUocHJlZGljdG9yLCByZXNwb25zZSkNCiAgZm9ybXVsYV9sbV8yIDwtIHJlZm9ybXVsYXRlKHNwcmludGYoInBvbHkoJXMsIDIpIiwgcHJlZGljdG9yKSwgcmVzcG9uc2UpDQogIGZvcm11bGFfbG1fMyA8LSByZWZvcm11bGF0ZShzcHJpbnRmKCJwb2x5KCVzLCAzKSIsIHByZWRpY3RvciksIHJlc3BvbnNlKQ0KICANCiAgIyBGaXQgYW5kIHN0b3JlIG1vZGVscyBpbiB0aGUgbGlzdA0KICBtb2RlbHNbW3Bhc3RlMChpZF9pLCAiX2xtXzAiKV1dIDwtIGJybSgNCiAgICBiZihmb3JtdWxhX2xtXzAsIGZhbWlseSA9IGdhdXNzaWFuKCkpLCANCiAgICBkYXRhID0gZGZfaSwgY29yZXMgPSA0LCBzZWVkID0gc2VlZF9udW1iZXIsIHNhdmVfcGFycyA9IHNhdmVfcGFycyhhbGwgPSBzYXZlX21vZGVscyksDQogICAgc2FtcGxlX3ByaW9yID0gRkFMU0UsIHNpbGVudCA9IFRSVUUsIGZpbGUgPSBwYXN0ZTAobW9kX291dHB1dF93ZCwgIi8iLCBpZF9pLCAiX2xtXzAiKQ0KICApDQogIA0KICBtb2RlbHNbW3Bhc3RlMChpZF9pLCAiX2xtXzEiKV1dIDwtIGJybSgNCiAgICBiZihmb3JtdWxhX2xtXzEsIGZhbWlseSA9IGdhdXNzaWFuKCkpLCANCiAgICBkYXRhID0gZGZfaSwgY29yZXMgPSA0LCBzZWVkID0gc2VlZF9udW1iZXIsIHNhdmVfcGFycyA9IHNhdmVfcGFycyhhbGwgPSBzYXZlX21vZGVscyksDQogICAgc2FtcGxlX3ByaW9yID0gRkFMU0UsIHNpbGVudCA9IFRSVUUsIGZpbGUgPSBwYXN0ZTAobW9kX291dHB1dF93ZCwgIi8iLCBpZF9pLCAiX2xtXzEiKQ0KICApDQogIA0KICBtb2RlbHNbW3Bhc3RlMChpZF9pLCAiX2xtXzIiKV1dIDwtIGJybSgNCiAgICBiZihmb3JtdWxhX2xtXzIsIGZhbWlseSA9IGdhdXNzaWFuKCkpLCANCiAgICBkYXRhID0gZGZfaSwgY29yZXMgPSA0LCBzZWVkID0gc2VlZF9udW1iZXIsIHNhdmVfcGFycyA9IHNhdmVfcGFycyhhbGwgPSBzYXZlX21vZGVscyksDQogICAgc2FtcGxlX3ByaW9yID0gRkFMU0UsIHNpbGVudCA9IFRSVUUsIGZpbGUgPSBwYXN0ZTAobW9kX291dHB1dF93ZCwgIi8iLCBpZF9pLCAiX2xtXzIiKQ0KICApDQogIA0KICBtb2RlbHNbW3Bhc3RlMChpZF9pLCAiX2xtXzMiKV1dIDwtIGJybSgNCiAgICBiZihmb3JtdWxhX2xtXzMsIGZhbWlseSA9IGdhdXNzaWFuKCkpLCANCiAgICBkYXRhID0gZGZfaSwgY29yZXMgPSA0LCBzZWVkID0gMTQzMDE5LCBzYXZlX3BhcnMgPSBzYXZlX3BhcnMoYWxsID0gc2F2ZV9tb2RlbHMpLA0KICAgIHNhbXBsZV9wcmlvciA9IEZBTFNFLCBzaWxlbnQgPSBUUlVFLCBmaWxlID0gcGFzdGUwKG1vZF9vdXRwdXRfd2QsICIvIiwgaWRfaSwgIl9sbV8zIikNCiAgKQ0KICANCiAgIyBSZXR1cm4gdGhlIGxpc3Qgb2YgbW9kZWxzIGZvciB0aGUgY3VycmVudCBJRA0KICByZXR1cm4obW9kZWxzKQ0KfQ0KYGBgDQoNCmBsb2FkX3JkczkoKWA6IEEgY3VzdG9tIGZ1bmN0aW9uIHRvIGxvYWQgYWxsIHJkcyBtb2RlbHMgaW4gYSBkaXJlY3RvcnkgYW5kIHN0b3JlIGluIGEgbGlzdC4gVGFrZXMgYSBkaXJlY3Rvcnkgd2l0aCBgLnJkc2AgZmlsZXMNCg0KYGBge3J9DQpsb2FkX3JkcyA8LSBmdW5jdGlvbihtb2RlbF9kdykgew0KICAjIExpc3QgYWxsIC5yZHMgZmlsZXMgaW4gdGhlIGRpcmVjdG9yeQ0KICBtb2RlbF9maWxlX2xpc3QgPC0gbGlzdC5maWxlcyhwYXRoID0gbW9kZWxfZHcsIHBhdHRlcm4gPSAiXFwucmRzJCIsIGZ1bGwubmFtZXMgPSBUUlVFKQ0KICANCiAgIyBTdHJpcCBleHRlbnNpb25zIGFuZCBnZXQgbmFtZXMNCiAgbW9kZWxfbmFtZXMgPC0gdG9vbHM6OmZpbGVfcGF0aF9zYW5zX2V4dChiYXNlbmFtZShtb2RlbF9maWxlX2xpc3QpKQ0KICANCiAgIyBSZWFkIGFsbCAucmRzIGZpbGVzIGludG8gYSBuYW1lZCBsaXN0DQogIG1vZGVsX3N0b3JlX2xpc3QgPC0gc2V0TmFtZXMobGFwcGx5KG1vZGVsX2ZpbGVfbGlzdCwgcmVhZFJEUyksIG1vZGVsX25hbWVzKQ0KICANCiAgcmV0dXJuKG1vZGVsX3N0b3JlX2xpc3QpDQp9DQpgYGANCg0KYGluY3JlbWVudGFsX3JlZ3Jlc3Npb25fYmF5ZXNfZml0cygpYDogQSBjdXN0b20gZnVuY3Rpb24gZm9yIHB1bGxpbmcgbW9kZWwgZml0cywgbG9vIGFuZCByMiB1c2luZyBgbG9vKClgIGFuZCBgYmF5ZXNfUjIoKWAsIHJlc3BlY3RpdmVseS4gVGFrZXMgYSBsaXN0IG9mIGBicm1gIG1vZGVscy4NCg0KYGBge3J9DQojIERlZmluZSBGdW5jdGlvbiB0byBQcm9jZXNzIHRoZSBkYXRhIGZvciBlYWNoIElEDQppbmNyZW1lbnRhbF9yZWdyZXNzaW9uX2JheWVzX2ZpdHMgPC0gZnVuY3Rpb24obW9kZWxzKSB7DQogIA0KICBsb29fcmVzdWx0c19saXN0IDwtIGxpc3QoKQ0KICANCiAgIyBJdGVyYXRlIG92ZXIgdGhlIG5hbWVzIG9mIHRoZSBtb2RlbHMNCiAgZm9yIChtb2RfbmFtZSBpbiBuYW1lcyhtb2RlbHMpKSB7DQogICAgIyBFeHRyYWN0IHRoZSBtb2RlbA0KICAgIG1vZF9pIDwtIG1vZGVsc1tbbW9kX25hbWVdXQ0KICAgIA0KICAgICMgQ29tcHV0ZSBMT08gcmVzdWx0cw0KICAgIG1vZF9sb29fcmVzdWx0c19pIDwtIGxvbzo6bG9vKG1vZF9pKQ0KICAgIA0KICAgICMgRXh0cmFjdCByZWxldmFudCBMT08gbWV0cmljcw0KICAgIGVscGRfbG9vX2kgPC0gbW9kX2xvb19yZXN1bHRzX2kkZWxwZF9sb28NCiAgICBwX2xvb19pIDwtIG1vZF9sb29fcmVzdWx0c19pJHBfbG9vDQogICAgbG9vaWNfaSA8LSBtb2RfbG9vX3Jlc3VsdHNfaSRsb29pYw0KICAgIA0KICAgICMgQ3JlYXRlIGEgZGF0YSBmcmFtZSB3aXRoIG1ldHJpY3MNCiAgICBkZl9pIDwtIGRhdGEuZnJhbWUoDQogICAgICBlbHBkX2xvbyA9IGVscGRfbG9vX2ksDQogICAgICBwX2xvbyA9IHBfbG9vX2ksDQogICAgICBsb29pYyA9IGxvb2ljX2ksDQogICAgICBtb2RlbCA9IG1vZF9uYW1lDQogICAgKQ0KICAgIA0KICAgIGVzdF9pIDwtIHRpZHkobW9kX2ksIGVmZmVjdHMgPSAiZml4ZWQiLCBjb25mLmludCA9IFRSVUUpICU+JSANCiAgICAgIGRwbHlyOjpzZWxlY3QodGVybSwgZXN0aW1hdGUsIGNvbmYubG93LCBjb25mLmhpZ2gpICU+JSANCiAgICAgIHRpZHlyOjpwaXZvdF93aWRlcigNCiAgICAgICAgbmFtZXNfZnJvbSA9IHRlcm0sICAgICAgICAgICAgICAgICMgVXNlIGB0ZXJtYCBhcyBjb2x1bW4gbmFtZXMNCiAgICAgICAgdmFsdWVzX2Zyb20gPSBjKGVzdGltYXRlLCBjb25mLmxvdywgY29uZi5oaWdoKSwgICMgVmFsdWVzIHRvIHBpdm90DQogICAgICAgIG5hbWVzX3NlcCA9ICJfIiAgICAgICAgICAgICAgICAgICAjIEFkZCBhIHNlcGFyYXRvciB0byBjb2x1bW4gbmFtZXMNCiAgICAgICkNCiAgICANCiAgICBkZl9pIDwtIGNiaW5kKGRmX2ksIGVzdF9pKQ0KICAgIA0KICAgICMgU3RvcmUgdGhlIGRhdGEgZnJhbWUgaW4gdGhlIGxpc3QNCiAgICBsb29fcmVzdWx0c19saXN0W1ttb2RfbmFtZV1dIDwtIGRmX2kNCiAgfQ0KICANCiAgIyBDb21iaW5kIA0KICBsb29fcmVzdWx0c19jb21iaW5lZCA8LSBiaW5kX3Jvd3MobG9vX3Jlc3VsdHNfbGlzdCkNCiAgDQogICMgR2V0IFIyIA0KICByMl9yZXN1bHRzIDwtIG1hcF9kZnIobW9kZWxzLCB+IGFzLmRhdGEuZnJhbWUoYmF5ZXNfUjIoLngpKSwgLmlkID0gIm1vZGVsIikgJT4lDQogICAgdGliYmxlOjpyZW1vdmVfcm93bmFtZXMoKQ0KICANCiAgIyBDb21iaW5kIFIyIGFuZCBsb28gcmVzdWx0cyANCiAgbW9kZWxfZml0X2RmIDwtIGRwbHlyOjpmdWxsX2pvaW4obG9vX3Jlc3VsdHNfY29tYmluZWQsIHIyX3Jlc3VsdHMsIGJ5ID0gIm1vZGVsIikgJT4lIA0KICAgIGRwbHlyOjpzZWxlY3QobW9kZWwsIGV2ZXJ5dGhpbmcoKSkgJT4lIA0KICAgIGRwbHlyOjpyZW5hbWUocjIgPSBFc3RpbWF0ZSwNCiAgICAgICAgICAgICAgICAgIHIyX2Vycm9yID0gRXN0LkVycm9yLA0KICAgICAgICAgICAgICAgICAgcjJfcTIuNSA9IFEyLjUsDQogICAgICAgICAgICAgICAgICByMl9xOTcuNSA9IFE5Ny41KSAlPiUgDQogICAgZHBseXI6Om11dGF0ZShpZCA9IHN1YigiXyhsbV9cXGQrKSQiLCAiIiwgbW9kZWwpLA0KICAgICAgICAgICAgICAgICAgbW9kZWxfdHlwZSA9IHN1YigiXi4qXyhsbV9cXGQrKSQiLCAiXFwxIiwgbW9kZWwpKQ0KICANCiAgcmV0dXJuKG1vZGVsX2ZpdF9kZikNCn0NCmBgYA0KDQpgYmF5ZXNfbW9kX3ByZWRpY3Rpb25zKClgOiBUaGlzIGZ1bmN0aW9uIGV4dHJhY3RzIHRoZSBwcmVkaWN0ZWQgdmFsdWVzIHVzaW5nIGBmaXR0ZWQoKWAgZnJvbSBhIGxpc3Qgb2YgbW9kZWxzIGFuZCBjb21iaW5lcyB0aGVtIHdpdGggdGhlIG9yaWdpbmFsIGRhdGEgZmlsZSB1c2VkIGZvciB0aGUgbW9kZWwuIFRoZXNlIGFyZSB0aGUgKipwb3N0ZXJpb3IgbWVhbiBmaXR0ZWQgdmFsdWVzKiogKGkuZS4gdGhlIGV4cGVjdGVkIHZhbHVlIG9mIHRoZSByZXNwb25zZSB2YXJpYWJsZSBnaXZlbiB0aGUgcHJlZGljdG9yIHZhcmlhYmxlcyBhbmQgdGhlIGVzdGltYXRlZCBwb3N0ZXJpb3IgZGlzdHJpYnV0aW9ucyBvZiB0aGUgcGFyYW1ldGVycykgZm9yIGVhY2ggb2JzZXJ2YXRpb24gaW4gdGhlIGRhdGFzZXQsIGFsb25nIHdpdGggKio5NSUgY3JlZGlibGUgaW50ZXJ2YWxzKiouDQoNCmBgYHtyfQ0KYmF5ZXNfbW9kX3ByZWRpY3Rpb25zIDwtIGZ1bmN0aW9uKG1vZGVscywgb3JpZ2luYWxfZGF0YSkgew0KICANCiAgcHJlZGljdGlvbl9saXN0IDwtIGxpc3QoKQ0KICANCiAgZm9yIChtb2RfbmFtZSBpbiBuYW1lcyhtb2RlbHMpKSB7DQogICAgIyBFeHRyYWN0IG1vZA0KICAgIG1vZF9pIDwtIG1vZGVsc1tbbW9kX25hbWVdXQ0KICAgIA0KICAgICMgTWFrZSBtb2RlIHByZWRpY3Rpb25zDQogICAgbW9kZWxfcHJlZGljdGlvbnNfaSA8LSBhcy5kYXRhLmZyYW1lKGZpdHRlZChtb2RfaSwgc3VtbWFyeSA9IFRSVUUpKSAlPiUgDQogICAgICBkcGx5cjo6bXV0YXRlKG1vZGVsID0gbW9kX25hbWUsDQogICAgICAgICAgICAgICAgICAgIGlkID0gc3ViKCJfKGxtX1xcZCspJCIsICIiLCBtb2RfbmFtZSksDQogICAgICAgICAgICAgICAgICAgIG1vZGVsX3R5cGUgPSBzdWIoIl4uKl8obG1fXFxkKykkIiwgIlxcMSIsIG1vZF9uYW1lKSkgJT4lIA0KICAgICAgZHBseXI6OnJlbmFtZShwcmVkX2xvd2VyID0gUTIuNSwgcHJlZF91cHBlciA9IFE5Ny41LCBwcmVkaWN0ZWQgPSBFc3RpbWF0ZSwgcHJlZF9lcnJvciA9IEVzdC5FcnJvcikgJT4lIA0KICAgICAgZHBseXI6OnNlbGVjdChtb2RlbCwgZXZlcnl0aGluZygpKQ0KICAgIA0KICAgIGlkX2kgPC0gbW9kZWxfcHJlZGljdGlvbnNfaSRpZFsxXQ0KICAgIA0KICAgIG9yaWdpbmFsX2RhdGFfaSA8LSBvcmlnaW5hbF9kYXRhICU+JSANCiAgICAgIGRwbHlyOjpmaWx0ZXIoaWQgPT0gaWRfaSkgJT4lIA0KICAgICAgZHBseXI6OnNlbGVjdCgtaWQpDQogICAgDQogICAgbW9kZWxfcHJlZGljdGlvbnNfb3JpZ2luYWxfaSA8LSBjYmluZChtb2RlbF9wcmVkaWN0aW9uc19pLCBvcmlnaW5hbF9kYXRhX2kpDQogICAgDQogICAgcHJlZGljdGlvbl9saXN0W1ttb2RfbmFtZV1dIDwtIG1vZGVsX3ByZWRpY3Rpb25zX29yaWdpbmFsX2kNCiAgfQ0KICBwcmVkaWN0aW9uc19kZiA8LSBiaW5kX3Jvd3MocHJlZGljdGlvbl9saXN0KSANCiAgcmV0dXJuKHByZWRpY3Rpb25zX2RmKQ0KfQ0KYGBgDQoNCmBjYWxjU01SKClgOiAqKmF1dGhvcmVkIGJ5IERlbmlzIENoYWJvdCoqIHVzZWQgdG8gZXN0aW1hdGUgU01SIHdpdGggc2V2ZXJhbCBkaWZmZXJlbnQgbWV0aG9kcyBDbGFpcmVhdXggYW5kIENoYWJvdCAoMjAxNikgXlsxXV4NCg0KXlsxXV4gQ2xhaXJlYXV4LCBHLiBhbmQgQ2hhYm90LCBELiAoMjAxNikgUmVzcG9uc2VzIGJ5IGZpc2hlcyB0byBlbnZpcm9ubWVudGFsIGh5cG94aWE6IGludGVncmF0aW9uIHRocm91Z2ggRnJ5J3MgY29uY2VwdCBvZiBhZXJvYmljIG1ldGFib2xpYyBzY29wZS4gSm91cm5hbCBvZiBGaXNoIEJpb2xvZ3kgPGh0dHBzOi8vZG9pLm9yZy8xMC4xMTExL2pmYi4xMjgzMz4NCg0KYGBge3J9DQpjYWxjU01SID0gZnVuY3Rpb24oWSwgcT1jKDAuMSwwLjE1LDAuMiwwLjI1LDAuMyksIEc9MTo0KXsNCgl1ID0gc29ydChZKQ0KCXRoZS5NY2x1c3QgPC0gTWNsdXN0KFksICBHPUcpDQoJY2wgPC0gdGhlLk1jbHVzdCRjbGFzc2lmaWNhdGlvbg0KCSMgc29tZXRpbWVzLCB0aGUgY2xhc3MgY29udGFpbmluZyBTTVIgaXMgbm90IGNhbGxlZCAxDQoJIyB0aGUgZm9sbG93aW5nIHByZXN1bWVzIHRoYXQgd2hlbiBjbGFzcyAxIGNvbnRhaW5zID4gMTAlIG9mIGNhc2VzLCANCgkjIGl0IGNvbnRhaW5zIFNNUiwgb3RoZXJ3aXNlIHdlIHRha2UgY2xhc3MgMg0KCWNsMiA8LSBhcy5kYXRhLmZyYW1lKHRhYmxlKGNsKSkNCgljbDIkY2wgPC0gYXMubnVtZXJpYyhsZXZlbHMoY2wyJGNsKSkNCgl2YWxpZCA8LSBjbDIkRnJlcT49MC4xKmxlbmd0aCh0aW1lKSAgDQoJdGhlLmNsIDwtIG1pbihjbDIkY2xbdmFsaWRdKQ0KCWxlZnQuZGlzdHIgPC0gWVt0aGUuTWNsdXN0JGNsYXNzaWZpY2F0aW9uPT10aGUuY2xdDQoJbWxuZCA9IHRoZS5NY2x1c3QkcGFyYW1ldGVycyRtZWFuW3RoZS5jbF0NCglDVm1sbmQgPSBzZChsZWZ0LmRpc3RyKS9tbG5kICogMTAwDQoJcXVhbnQ9cXVhbnRpbGUoWSwgcSkNCglsb3cxMD1tZWFuKHVbMToxMF0pDQoJbG93MTBwYyA9IG1lYW4odVs2Oig1ICsgcm91bmQoMC4xKihsZW5ndGgodSktNSkpKV0pDQoJIyByZW1vdmUgNSBvdXRsaWVycywga2VlcCBsb3dlc3QgMTAlIG9mIHRoZSByZXN0LCBhdmVyYWdlDQoJIyBIZXJybWFubiAmIEVuZGVycyAyMDAwDQoJcmV0dXJuKGxpc3QobWxuZD1tbG5kLCBxdWFudD1xdWFudCwgbG93MTA9bG93MTAsIGxvdzEwcGM9bG93MTBwYywNCgkJICAgICAgY2w9Y2wsIENWbWxuZD1DVm1sbmQpKQ0KfQ0KYGBgDQoNCmBjYWxjTzJjcml0KClgOiAqKmF1dGhvcmVkIGJ5IERlbmlzIENoYWJvdCoqIHVzZWQgdG8gZXN0aW1hdGUgTzJjcml0IChQY3JpcHQpLiBDbGFpcmVhdXggYW5kIENoYWJvdCAoMjAxNikgXlsxXV4NCg0KKioqTm90ZTogTzIgaXMgYXNzdW1lZCB0byBiZSBpbiBwZXJjZW50YWdlIG9mIGRpc3NvbHZlZCBveHlnZW4gKERPKSoqKg0KDQpgYGB7cn0NCmNhbGNPMmNyaXQgPC0gZnVuY3Rpb24oRGF0YSwgU01SLCBsb3dlc3RNTzIgPSBOQSwgZ2FwTGltaXQgPSA0LA0KbWF4Lm5iLk1PMi5mb3IucmVnID0gMjApDQp7DQojIEFVVEhPUjogRGVuaXMgQ2hhYm90LCBJbnN0aXR1dCBNYXVyaWNlLUxhbW9udGFnbmUsIERGTywgQ2FuYWRhDQojIGZpcnN0IHZlcnNpb24gd3JpdHRlbiBpbiBKdW5lIDIwMDkNCiMgbGFzdCB1cGRhdGVkIGluIEphbnVhcnkgMjAxNQ0KbWV0aG9kID0gIkxTX3JlZyIgIyB3aWxsIGJlY29tZSAidGhyb3VnaF9vcmlnaW4iIGlmIGludGVyY2VwdCBpcyA+IDANCmlmKGlzLm5hKGxvd2VzdE1PMikpIGxvd2VzdE1PMiA9IHF1YW50aWxlKERhdGEkTU8yW0RhdGEkRE8gPj0gODBdLCBwPTAuMDUpDQojIFN0ZXAgMTogaWRlbnRpZnkgcG9pbnRzIHdoZXJlIE1PMiBpcyBwcm9wb3J0aW9uYWwgdG8gRE8NCmdlcVNNUiA9IERhdGEkTU8yID49IGxvd2VzdE1PMg0KcGl2b3RETyA9IG1pbihEYXRhJERPW2dlcVNNUl0pDQpsZXRoYWwgPSBEYXRhJERPIDwgcGl2b3RETw0KTl91bmRlcl9TTVIgPSBzdW0obGV0aGFsKSAjIHBvaW50cyBhdmFpbGFibGUgZm9yIHJlZ3Jlc3Npb24/DQpmaW5hbF9OX3VuZGVyX1NNUiA9IGxldGhhbCAjIHNvbWUgcG9pbnRzIG1heSBiZSByZW1vdmVkIGF0IFN0ZXAgNA0KbGFzdE1PMnJlZyA9IERhdGEkTU8yW0RhdGEkRE8gPT0gcGl2b3RET10gIyBsYXN0IE1PMiB3aGVuIHJlZ3VsYXRpbmcNCmlmKE5fdW5kZXJfU01SID4gMSkgdGhlTW9kID0gbG0oTU8yfkRPLCBkYXRhPURhdGFbbGV0aGFsLF0pDQojIFN0ZXAgMiwgYWRkIG9uZSBvciBtb3JlIHBvaW50IGF0IG9yIGFib3ZlIFNNUg0KIyAyQSwgd2hlbiB0aGVyZSBhcmUgZmV3ZXIgdGhhbiAzIHZhbGlkIHBvaW50cyB0byBjYWxjdWxhdGUgYSByZWdyZXNzaW9uDQppZihOX3VuZGVyX1NNUiA8IDMpew0KbWlzc2luZyA9IDMgLSBzdW0obGV0aGFsKQ0Kbm90LmxldGhhbCA9IERhdGEkRE9bZ2VxU01SXQ0KRE9saW1pdCA9IG1heChzb3J0KG5vdC5sZXRoYWwpWzE6bWlzc2luZ10pICMgaGlnaGVzdCBETyBhY2NlcHRhYmxlDQojIHRvIHJlYWNoIGEgTiBvZiAzDQphZGRlZFBvaW50cyA9IERhdGEkRE8gPD0gRE9saW1pdA0KbGV0aGFsID0gbGV0aGFsIHwgYWRkZWRQb2ludHMNCnRoZU1vZCA9IGxtKE1PMn5ETywgZGF0YT1EYXRhW2xldGhhbCxdKQ0KfQ0KIyAyQiwgYWRkIHBpdm90RE8gdG8gdGhlIGZpdCB3aGVuIFN0ZXAgMSB5aWVsZGVkIDMgb3IgbW9yZSB2YWx1ZXM/DQppZihOX3VuZGVyX1NNUiA+PSAzKXsNCmxldGhhbEIgPSBEYXRhJERPIDw9IHBpdm90RE8gIyBoYXMgb25lIG1vcmUgdmFsdWUgdGhhbiAibGV0aGFsIg0KcmVnQSA9IHRoZU1vZA0KcmVnQiA9IGxtKE1PMn5ETywgZGF0YT1EYXRhW2xldGhhbEIsXSkNCmxhcmdlX3Nsb3BlX2Ryb3AgPSAoY29lZihyZWdBKVsyXS9jb2VmKHJlZ0IpWzJdKSA+IDEuMSAjIGFyYml0cmFyeQ0KbGFyZ2VfRE9fZ2FwID0gKG1heChEYXRhJERPW2xldGhhbEJdKSAtIG1heChEYXRhJERPW2xldGhhbF0pKSA+IGdhcExpbWl0DQp0b29TbWFsbE1PMiA9IGxhc3RNTzJyZWcgPCBTTVINCmlmKCFsYXJnZV9zbG9wZV9kcm9wICYgIWxhcmdlX0RPX2dhcCAmICF0b29TbWFsbE1PMikgew0KbGV0aGFsID0gbGV0aGFsQg0KdGhlTW9kID0gcmVnQg0KfSAjIG90aGVyd2lzZSB3ZSBkbyBub3QgYWNjZXB0IHRoZSBhZGRpdGlvbmFsIHBvaW50DQp9DQojIFN0ZXAgMw0KIyBpZiB0aGUgdXNlciB3YW50cyB0byBsaW1pdCB0aGUgbnVtYmVyIG9mIHBvaW50cyBpbiB0aGUgcmVncmVzc2lvbg0KaWYoIWlzLm5hKG1heC5uYi5NTzIuZm9yLnJlZykgJiBzdW0obGV0aGFsKT5tYXgubmIuTU8yLmZvci5yZWcpew0KUmFua3MgPSByYW5rKERhdGEkRE8pDQpsZXRoYWwgPSBSYW5rcyA8PSBtYXgubmIuTU8yLmZvci5yZWcNCnRoZU1vZCA9IGxtKE1PMn5ETywgZGF0YT1EYXRhW2xldGhhbCxdKQ0KZmluYWxfTl91bmRlcl9TTVIgPSBtYXgubmIuTU8yLmZvci5yZWcNCn0NCiMgU3RlcCA0DQpwcmVkTU8yID0gYXMubnVtZXJpYyhwcmVkaWN0KHRoZU1vZCwgZGF0YS5mcmFtZShETz1EYXRhJERPKSkpDQpEYXRhJGRlbHRhID0gKERhdGEkTU8yLXByZWRNTzIpL3ByZWRNTzIgKiAxMDAgIyByZXNpZHVhbHMgc2V0IHRvIHplcm8NCiMgd2hlbiBiZWxvdyBwaXZvdERPDQpEYXRhJGRlbHRhW0RhdGEkRE8gPCBwaXZvdERPIHwgbGV0aGFsXSA9IDANCnRvbCA9IDAgIyBhbnkgcG9zaXRpdmUgcmVzaWR1YWwgaXMgdW5hY2NlcHRhYmxlDQpIaWdoVmFsdWVzID0gRGF0YSRkZWx0YSA+IHRvbA0KUmFua3MgPSByYW5rKC0xKkRhdGEkZGVsdGEpDQpIaWdoTU8yID0gSGlnaFZhbHVlcyAmIFJhbmtzID09IG1pbihSYW5rcykgIyBrZWVwIGxhcmdlc3QgcmVzaWR1YWwNCmlmIChzdW0oSGlnaFZhbHVlcykgPiAwKSB7DQpuYmxldGhhbCA9IHN1bShsZXRoYWwpDQpEYXRhJFcgPSBOQQ0KRGF0YSRXW2xldGhhbF09MS9uYmxldGhhbA0KRGF0YSRXW0hpZ2hNTzJdID0gMQ0KdGhlTW9kID0gbG0oTU8yfkRPLCB3ZWlnaHQ9VywgZGF0YT1EYXRhW2xldGhhbCB8IEhpZ2hNTzIsXSkNCiMgVGhpcyBuZXcgcmVncmVzc2lvbiBpcyBhbHdheXMgYW4gaW1wcm92ZW1lbnQsIGJ1dCB0aGVyZSBjYW4gc3RpbGwNCiMgYmUgcG9pbnRzIGFib3ZlIHRoZSBsaW5lLCBzbyB3ZSByZXBlYXQNCnByZWRNTzJfMiA9IGFzLm51bWVyaWMocHJlZGljdCh0aGVNb2QsIGRhdGEuZnJhbWUoRE89RGF0YSRETykpKQ0KRGF0YSRkZWx0YTIgPSAoRGF0YSRNTzItcHJlZE1PMl8yKS9wcmVkTU8yXzIgKiAxMDANCkRhdGEkZGVsdGEyW0RhdGEkRE8gPCBwaXZvdERPXSA9IDANCnRvbCA9IERhdGEkZGVsdGEyW0hpZ2hNTzJdDQpIaWdoVmFsdWVzMiA9IERhdGEkZGVsdGEyID4gdG9sDQppZihzdW0oSGlnaFZhbHVlczIpPjApew0KUmFua3MyID0gcmFuaygtMSpEYXRhJGRlbHRhMikNCkhpZ2hNTzJfMiA9IEhpZ2hWYWx1ZXMyICYgUmFua3MyID09IDEgIyBrZWVwIHRoZSBsYXJnZXN0IHJlc2lkdWFsDQpuYmxldGhhbCA9IHN1bShsZXRoYWwpDQpEYXRhJFcgPSBOQQ0KRGF0YSRXW2xldGhhbF09MS9uYmxldGhhbA0KRGF0YSRXW0hpZ2hNTzJfMl0gPSAxDQp0aGVNb2QyID0gbG0oTU8yfkRPLCB3ZWlnaHQ9VywgZGF0YT1EYXRhW2xldGhhbCB8IEhpZ2hNTzJfMixdKQ0KIyBpcyBuZXcgc2xvcGUgc3RlZXBlciB0aGFuIHRoZSBvbGQgb25lPw0KaWYodGhlTW9kMiRjb2VmWzJdID4gdGhlTW9kJGNvZWZbMl0pIHsNCnRoZU1vZCA9IHRoZU1vZDINCkhpZ2hNTzIgPSBIaWdoTU8yXzINCn0NCn0gIyBlbmQgc2Vjb25kIHNlYXJjaCBmb3IgaGlnaCB2YWx1ZQ0KfSAjIGVuZCBmaXJzdCBzZWFyY2ggZm9yIGhpZ2ggdmFsdWUNCkNvZWYgPSBjb2VmZmljaWVudHModGhlTW9kKQ0KI1N0ZXAgNSwgY2hlY2sgZm9yIHBvc2l0aXZlIGludGVyY2VwdA0KQWJvdmVPcmlnaW4gPSBDb2VmWzFdID4gMA0KIyBpZiBpdCBpcywgd2UgdXNlIGEgcmVncmVzc2lvbiB0aGF0IGdvZXMgdGhyb3VnaCB0aGUgb3JpZ2luDQppZiAoQWJvdmVPcmlnaW4pew0KdGhlTW9kID0gbG0oTU8yfkRPIC0xLCBkYXRhPURhdGFbbGV0aGFsLF0pDQpDb2VmID0gYygwLCBjb2VmZmljaWVudHModGhlTW9kKSkgIyBuZWVkIHRvIGFkZCB0aGUgaW50ZXJjZXB0ICgwKQ0KIyBtYW51YWxseSB0byBoYXZlIGEgcGFpciBvZiBjb2VmZmljaWVudHMNCm1ldGhvZCA9ICJ0aHJvdWdoX29yaWdpbiINCkhpZ2hNTzIgPSByZXAoRkFMU0UsIG5yb3coRGF0YSkpICMgZGlkIG5vdCB1c2UgdGhlIGFkZGl0aW9uYWwgdmFsdWUNCiMgZnJvbSBTdGVwIDQNCn0NCnBvMmNyaXQgPSBhcy5udW1lcmljKHJvdW5kKChTTVIgLSBDb2VmWzFdKSAvIENvZWZbMl0sIDEpKQ0Kc3VtX21vZCA9IHN1bW1hcnkodGhlTW9kKQ0KYW5vdl9tb2QgPSBhbm92YSh0aGVNb2QpDQpPMkNSSVQgPSBsaXN0KG8yY3JpdD1wbzJjcml0LCBTTVI9U01SLCBOYl9NTzJfY29uZm9ybWluZyA9IE5fdW5kZXJfU01SLA0KTmJfTU8yX2NvbmZfdXNlZCA9IGZpbmFsX05fdW5kZXJfU01SLA0KSGlnaF9NTzJfcmVxdWlyZWQgPSBzdW0oSGlnaE1PMikgPT0gMSwgb3JpZ0RhdGE9RGF0YSwNCk1ldGhvZD1tZXRob2QsIG1vZD10aGVNb2QsIHIyID0gc3VtX21vZCRyLnNxdWFyZWQsDQpQID0gYW5vdl9tb2QkIlByKD5GKSIsIGxldGhhbFBvaW50cyA9IHdoaWNoKGxldGhhbCksDQpBZGRlZFBvaW50cyA9IHdoaWNoKEhpZ2hNTzIpKQ0KfSAjIGVuZCBmdW5jdGlvbg0KYGBgDQoNCmBwbG90TzJjcml0KClgOiAqKmF1dGhvcmVkIGJ5IERlbmlzIENoYWJvdCoqLCB1c2VkIHRvIHBsb3QgdGhlIG1vZGVzIHVzZWQgZm9yIHRoZSBgY2FsY08yY3JpdCgpYCBmdW5jdGlvbi4gQ2xhaXJlYXV4IGFuZCBDaGFib3QgKDIwMTYpIF5bMV1eDQoNCioqKk5vdGU6IEkgYWRkZWQgYWJsaW5lKGg9bG93ZXN0TU8yLCBjb2w9InBpbmsiKSBzbyB0aGF0IEkgY291bGQgdmlzdWFsaXNlIHRoZSBsb3dlc3RNTzIgcG9zaXRpb24qKioNCg0KYGBge3J9DQpwbG90TzJjcml0IDwtIGZ1bmN0aW9uKG8yY3JpdG9iaiwgcGxvdElEPSIiLA0KWGxhYj0iRGlzc29sdmVkIG94eWdlbiAoJSBzYXQuKSIsIFlsYWI9ImRvdGl0YWx1bW9sIiwNCnNtci5jZXg9MC45LCBvMmNyaXQuY2V4PTAuOSwgcGxvdElELmNleD0xLjIsDQpUcmFuc3BhcmVuY3k9VCwuLi4pDQp7DQojIEFVVEhPUjogRGVuaXMgQ2hhYm90LCBJbnN0aXR1dCBNYXVyaWNlLUxhbW9udGFnbmUsIERGTywgQ2FuYWRhDQojIGZpcnN0IHZlcnNpb24gd3JpdHRlbiBpbiBKdW5lIDIwMDkNCiMgbGFzdCB1cGRhdGVkIDIwMTUtMDItMDkNCiMgZm9yIFIgcGxvdHRpbmcgZGV2aWNlcyB0aGF0IGRvIG5vdCBzdXBwb3J0IHRyYW5zcGFyZW5jeQ0KIyAoZS5nLiwgcG9zdHNjcmlwdCksIHNldCBUcmFuc3BhcmVuY3kgdG8gRkFMU0UNCnNtciA9IG8yY3JpdG9iaiRTTVINCmlmKFlsYWIgJWluJSBjKCJkb3RpdGFsdW1vbCIsICJpdGFsdW1vbCIsICJkb3R1bW9sIiwgInVtb2wiLA0KImRvdGl0YWxtZyIsICJpdGFsbWciLCAiZG90bWciLCAibWciKSkgew0Kc3dpdGNoKFlsYWIsDQpkb3RpdGFsdW1vbCA9IHsNCm1vMi5sYWIgPSBleHByZXNzaW9uKHBhc3RlKGl0YWxpYyhkb3QoTSkpW09bMl1dLCAiICgiLG11LCJtb2wgIiwgT1syXSwNCiIgIiwgbWluXi0xLCAiICIsIGtnXi0xLCAiKSIpKQ0KfSwNCml0YWx1bW9sID0gew0KbW8yLmxhYiA9IGV4cHJlc3Npb24ocGFzdGUoaXRhbGljKE0pW09bMl1dLCAiICgiLG11LCJtb2wgIiwgT1syXSwgIiAiLA0KbWluXi0xLCAiICIsIGtnXi0xLCAiKSIpKQ0KfSwNCmRvdHVtb2wgPSB7DQptbzIubGFiID0gZXhwcmVzc2lvbihwYXN0ZShkb3QoTSlbT1syXV0sICIgKCIsbXUsIm1vbCAiLCBPWzJdLCAiICIsDQptaW5eLTEsICIgIiwga2deLTEsICIpIikpDQp9LA0KdW1vbCA9IHsNCm1vMi5sYWIgPSBleHByZXNzaW9uKHBhc3RlKE1bT1syXV0sICIgKCIsbXUsIm1vbCAiLCBPWzJdLCAiICIsIG1pbl4tMSwNCiIgIiwga2deLTEsICIpIikpDQp9LA0KZG90aXRhbG1nID0gew0KbW8yLmxhYiA9IGV4cHJlc3Npb24ocGFzdGUoaXRhbGljKGRvdChNKSlbT1syXV0sICIgKG1nICIsIE9bMl0sICIgIiwNCmheLTEsICIgIiwga2deLTEsICIpIikpDQp9LA0KaXRhbG1nID0gew0KbW8yLmxhYiA9IGV4cHJlc3Npb24ocGFzdGUoaXRhbGljKE0pW09bMl1dLCAiIChtZyAiLCBPWzJdLCAiICIsDQpoXi0xLCAiICIsIGtnXi0xLCAiKSIpKQ0KfSwNCmRvdG1nID0gew0KbW8yLmxhYiA9IGV4cHJlc3Npb24ocGFzdGUoZG90KE0pW09bMl1dLCAiIChtZyAiLCBPWzJdLCAiICIsIGheLTEsICIgIiwNCmtnXi0xLCAiKSIpKQ0KfSwNCm1nID0gew0KbW8yLmxhYiA9IGV4cHJlc3Npb24ocGFzdGUoTVtPWzJdXSwgIiAobWcgIiwgT1syXSwgIiAiLCBoXi0xLCAiICIsDQprZ14tMSwgIikiKSkNCn0NCikNCn0gZWxzZSBtbzIubGFiPVlsYWINCmlmKFRyYW5zcGFyZW5jeSkge0NvbD1jKHJnYigwLDAsMCwwLjcpLCAicmVkIiwgIm9yYW5nZSIpDQp9IGVsc2Uge0NvbD1jKGdyZXkoMC4zKSwgInJlZCIsICJvcmFuZ2UiKX0NCkRhdGE9bzJjcml0b2JqJG9yaWdEYXRhDQpsb3dlc3RNTzIgPSBxdWFudGlsZShEYXRhJE1PMltEYXRhJERPID49IDgwXSwgcD0wLjA1KSAjIEkgYWRkZWQgdGhpcw0KRGF0YSRDb2xvciA9IENvbFsxXQ0KRGF0YSRDb2xvcltvMmNyaXRvYmokbGV0aGFsUG9pbnRzXSA9IENvbFsyXQ0KRGF0YSRDb2xvcltvMmNyaXRvYmokQWRkZWRQb2ludHNdID0gQ29sWzNdDQojIG9yZGluYXJ5IExTIHJlZ3Jlc3Npb24gd2l0aG91dCBhZGRlZCBwb2ludHM6IGJsdWUgbGluZSwgcmVkIHN5bWJvbHMNCiMgb3JkaW5hcnkgTFMgcmVncmVzc2lvbiB3aXRoIGFkZGVkIHBvaW50czogYmx1ZSBsaW5lLCByZWQgJiBvcmFuZ2Ugc3ltYm9scw0KIyByZWdyZXNzaW9uIHRocm91Z2ggb3JpZ2luOiBncmVlbiBkb3R0ZWQgbGluZSwgcmVkIHN5bWJvbHMNCmxpbmUuY29sb3IgPSBpZmVsc2UobzJjcml0b2JqJE1ldGhvZD09IkxTX3JlZyIsICJibHVlIiwgImRhcmtncmVlbiIpDQpsaW5lLnR5cGUgPSBpZmVsc2UobzJjcml0b2JqJE1ldGhvZD09IkxTX3JlZyIsIDEsIDMpDQpsaW1YID0gYygwLCBtYXgoRGF0YSRETykpDQpsaW1ZID0gYygwLCBtYXgoRGF0YSRNTzIpKQ0KcGxvdChNTzJ+RE8sIGRhdGE9RGF0YSwgeGxpbT1saW1YLCB5bGltPWxpbVksIGNvbD1EYXRhJENvbG9yLCB4bGFiPVhsYWIsDQp5bGFiPW1vMi5sYWIsIC4uLikNCmNvb3JkIDwtIHBhcigidXNyIikNCmlmKHBsb3RJRCAhPSAiIil7DQp0ZXh0KDAsIGNvb3JkWzRdLCBwbG90SUQsIGNleD1wbG90SUQuY2V4LCBhZGo9YygwLDEuMikpDQp9DQphYmxpbmUoaD1sb3dlc3RNTzIsIGNvbD0icGluayIpICMgSSBhZGRlZCB0aGlzDQphYmxpbmUoaD1zbXIsIGNvbD0ib3JhbmdlIikNCnRleHQoY29vcmRbMV0sIHNtciwgIlNNUiIsIGFkaj1jKC0wLjEsMS4zKSwgY2V4PXNtci5jZXgpDQp0ZXh0KGNvb3JkWzFdLCBzbXIsIHJvdW5kKHNtciwxKSwgYWRqPWMoLTAuMSwtMC4zKSwgY2V4PXNtci5jZXgpDQppZighaXMubmEobzJjcml0b2JqJG8yY3JpdCkpIHsNCmFibGluZShvMmNyaXRvYmokbW9kLCBjb2w9bGluZS5jb2xvciwgbHR5PWxpbmUudHlwZSkNCnNlZ21lbnRzKG8yY3JpdG9iaiRvMmNyaXQsIHNtciwgbzJjcml0b2JqJG8yY3JpdCwgY29vcmRbM10sDQpjb2w9bGluZS5jb2xvciwgbHdkPTEpDQp0ZXh0KHg9bzJjcml0b2JqJG8yY3JpdCwgeT0wLCBvMmNyaXRvYmokbzJjcml0LCBjb2w9bGluZS5jb2xvciwNCmNleD1vMmNyaXQuY2V4LCBhZGo9YygtMC4xLDAuNSkpDQp9DQp9ICMgZW5kIG9mIGZ1bmN0aW9uDQpgYGANCg0KPCEtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tPg0KDQojIPCfk4IgRGlyZWN0b3JpZXMNCg0KPCEtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tPg0KDQojIyBJbnB1dA0KDQoqKvCfk6UgaW5wdXRfZGF0YV93ZCoqOiBEaXJlY3RvcnkgZm9yIHRoZSBtZXRhZGF0YQ0KDQpgYGB7cn0NCndkIDwtIGdldHdkKCkNCmlucHV0X2RhdGFfd2QgPC0gcGFzdGUwKHdkLCAiLi9pbnB1dC1kYXRhIikgIyBjcmVhdGVzIGEgdmFyaWFibGUgd2l0aCB0aGUgbmFtZSBvZiB0aGUgd2Qgd2Ugd2FudCB0byB1c2UNCmBgYA0KDQoqKvCfk6UgbW9kX2RhdGFfd2QqKjogRGlyZWN0b3J5IGZvciBtb2RlbCBvdXRwdXQgZGF0YSBlc3RpbWF0ZWQgc2xvcGVzDQoNCmBgYHtyfQ0KbW9kX2RhdGFfd2QgPC0gcGFzdGUwKHdkLCAiLi9tb2QtZGF0YSIpDQpgYGANCg0KIyMgT3V0cHV0DQoNCioq8J+TpCBvdXRwdXRfZmlnX3dkKio6IHRoaXMgaXMgd2hlcmUgd2Ugd2lsbCBwdXQgdGhlIGZpZ3VyZXMNCg0KYGBge3J9DQpvdXRwdXRfZmlnX3dkIDwtIHBhc3RlMCh3ZCwgIi4vb3V0cHV0LWZpZyIpDQppZiAoIWRpci5leGlzdHMoIm91dHB1dC1maWciKSkge2Rpci5jcmVhdGUoIm91dHB1dC1maWciKX0NCmBgYA0KDQoqKvCfk6Qgb3V0cHV0X21vZHNfd2QqKjogdGhpcyBpcyB3aGVyZSB3ZSB3aWxsIHB1dCB0aGUgbW9kZWxzDQoNCmBgYHtyfQ0Kb3V0cHV0X21vZHNfd2QgPC0gcGFzdGUwKHdkLCAiLi9vdXRwdXQtbW9kIikNCmlmICghZGlyLmV4aXN0cygib3V0cHV0LW1vZCIpKSB7ZGlyLmNyZWF0ZSgib3V0cHV0LW1vZCIpfQ0KYGBgDQoNCjwhLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLT4NCg0KIyDwn5K/IERhdGENCg0KPCEtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tPg0KDQojIyBTbG9wZXMgKOG5gE8pDQoNCioq8J+SvyBzbG9wZV9kZioqOiBXZSBoYXZlIGltcG9ydGVkIHRoZSBzbG9wZXMgZXh0cmFjdGVkIGluIExhYkNoYXJ0IGR1cmluZyBlYWNoIHBoYXNlIG9mIHRoZSBleHBlcmltZW50DQoNCmBgYHtyfQ0KIHNldHdkKGlucHV0X2RhdGFfd2QpDQoNCmRhdGFfZmlsZSA8LSAibGFiY2hhcnQtc2xvcGUtZGF0YS54bHN4Ig0KDQojIA0KIyAjIEdldCB0aGUgbmFtZXMgb2YgYWxsIHNoZWV0cyBpbiB0aGUgRXhjZWwgZmlsZQ0Kc2hlZXRfbmFtZXMgPC0gcmVhZHhsOjpleGNlbF9zaGVldHMoZGF0YV9maWxlKQ0KYWxsX3RyaWFsc19zZWxlY3QgPC0gYygic3RhcnRfZGF0ZSIsICJvcmRlciIsICJwaGFzZSIsICJjeWNsZSIsICJkYXRlIiwgInRpbWUiKQ0Kc2xvcGVfbGlzdCA8LSBsaXN0KCkNCg0KZm9yIChzaGVldCBpbiBzaGVldF9uYW1lcykgew0KDQogIGRmIDwtIHJlYWRfZXhjZWwoZGF0YV9maWxlLCBzaGVldCA9IHNoZWV0KSAlPiUgDQogIGRwbHlyOjpyZW5hbWVfd2l0aCh0b2xvd2VyKQ0KICANCmFfbmFtZSA8LSBwYXN0ZTAoImFfIiwgdG9sb3dlcihzaGVldCkpDQphX2RmIDwtIGRmICU+JQ0KICBkcGx5cjo6c2VsZWN0KHN0YXJ0c193aXRoKCdhJyksIGFsbF90cmlhbHNfc2VsZWN0KSAlPiUgDQogIGRwbHlyOjpyZW5hbWUodGVtcCA9IGFfdGVtcCkgJT4lIA0KICBkcGx5cjo6bXV0YXRlKGFjcm9zcyhzdGFydHNfd2l0aCgnYScpLCBhcy5udW1lcmljKSkgJT4lIA0KICBwaXZvdF9sb25nZXIoDQogICAgY29scyA9IHN0YXJ0c193aXRoKCdhJyksICMgU2VsZWN0IGFsbCBjb2x1bW5zIHRvIHBpdm90DQogICAgbmFtZXNfdG8gPSBjKCJjaGFtYmVyX2lkIiwgIi52YWx1ZSIpLCAjIFNlcGFyYXRlIGNvbHVtbiBuYW1lcyBpbnRvICdpZCcgYW5kIG90aGVyIHZhcmlhYmxlcw0KICAgIG5hbWVzX3NlcCA9ICJfIg0KICApICU+JQ0KICBkcGx5cjo6bXV0YXRlKHJlc3Bpcm9tZXRlcl9ncm91cCA9ICJhIikgIyBBZGQgYSBuZXcgY29sdW1uIHdpdGggYSBmaXhlZCB2YWx1ZQ0KDQpzbG9wZV9saXN0W1thX25hbWVdXTwtIGFfZGYNCg0KYl9uYW1lIDwtIHBhc3RlMCgiYl8iLCB0b2xvd2VyKHNoZWV0KSkNCmJfZGYgPC0gZGYgJT4lIA0KICBkcGx5cjo6c2VsZWN0KHN0YXJ0c193aXRoKCdiJyksIGFsbF90cmlhbHNfc2VsZWN0KSAlPiUgDQogIGRwbHlyOjpyZW5hbWUodGVtcCA9IGJfdGVtcCkgJT4lIA0KICBkcGx5cjo6bXV0YXRlKGFjcm9zcyhzdGFydHNfd2l0aCgnYicpLCBhcy5udW1lcmljKSkgJT4lIA0KICBwaXZvdF9sb25nZXIoDQogICAgY29scyA9IHN0YXJ0c193aXRoKCdiJyksICMgU2VsZWN0IGFsbCBjb2x1bW5zIHRvIHBpdm90DQogICAgbmFtZXNfdG8gPSBjKCJjaGFtYmVyX2lkIiwgIi52YWx1ZSIpLCAjIFNlcGFyYXRlIGNvbHVtbiBuYW1lcyBpbnRvICdpZCcgYW5kIG90aGVyIHZhcmlhYmxlcw0KICAgIG5hbWVzX3NlcCA9ICJfIg0KICApICU+JSANCiAgICBkcGx5cjo6bXV0YXRlKHJlc3Bpcm9tZXRlcl9ncm91cCA9ICJiIikNCg0Kc2xvcGVfbGlzdFtbYl9uYW1lXV0gPC0gYl9kZg0KDQpjX25hbWUgPC0gcGFzdGUwKCJjXyIsIHRvbG93ZXIoc2hlZXQpKQ0KY19kZiA8LSBkZiAlPiUgDQogIGRwbHlyOjpzZWxlY3Qoc3RhcnRzX3dpdGgoJ2MnKSwgYWxsX3RyaWFsc19zZWxlY3QpICU+JSANCiAgZHBseXI6OnJlbmFtZSh0ZW1wID0gY190ZW1wLA0KICAgICAgICAgICAgICAgIGlfY3ljbGUgPSBjeWNsZSkgJT4lIA0KICBkcGx5cjo6bXV0YXRlKGFjcm9zcyhzdGFydHNfd2l0aCgnYycpLCBhcy5udW1lcmljKSkgJT4lDQogIHBpdm90X2xvbmdlcigNCiAgICBjb2xzID0gc3RhcnRzX3dpdGgoJ2MnKSwgIyBTZWxlY3QgYWxsIGNvbHVtbnMgdG8gcGl2b3QNCiAgICBuYW1lc190byA9IGMoImNoYW1iZXJfaWQiLCAiLnZhbHVlIiksICMgU2VwYXJhdGUgY29sdW1uIG5hbWVzIGludG8gJ2lkJyBhbmQgb3RoZXIgdmFyaWFibGVzDQogICAgbmFtZXNfc2VwID0gIl8iDQogICkgJT4lIA0KICAgIGRwbHlyOjptdXRhdGUocmVzcGlyb21ldGVyX2dyb3VwID0gImMiKSAlPiUgDQogIGRwbHlyOjpyZW5hbWUoY3ljbGUgPSBpX2N5Y2xlKQ0KDQpzbG9wZV9saXN0W1tjX25hbWVdXSA8LSBjX2RmDQoNCmRfbmFtZSA8LSBwYXN0ZTAoImRfIiwgdG9sb3dlcihzaGVldCkpDQpkX2RmIDwtIGRmICU+JSANCiAgZHBseXI6OnNlbGVjdChzdGFydHNfd2l0aCgnZCcpLCBhbGxfdHJpYWxzX3NlbGVjdCkgJT4lIA0KICBkcGx5cjo6cmVuYW1lKHRlbXAgPSBkX3RlbXAsDQogICAgICAgICAgICAgICAgaV9kYXRlID0gZGF0ZSkgJT4lIA0KICBkcGx5cjo6bXV0YXRlKGFjcm9zcyhzdGFydHNfd2l0aCgnZCcpLCBhcy5udW1lcmljKSkgJT4lDQogIHBpdm90X2xvbmdlcigNCiAgICBjb2xzID0gc3RhcnRzX3dpdGgoJ2QnKSwgIyBTZWxlY3QgYWxsIGNvbHVtbnMgdG8gcGl2b3QNCiAgICBuYW1lc190byA9IGMoImNoYW1iZXJfaWQiLCAiLnZhbHVlIiksICMgU2VwYXJhdGUgY29sdW1uIG5hbWVzIGludG8gJ2lkJyBhbmQgb3RoZXIgdmFyaWFibGVzDQogICAgbmFtZXNfc2VwID0gIl8iDQogICkgJT4lIA0KICAgIGRwbHlyOjptdXRhdGUocmVzcGlyb21ldGVyX2dyb3VwID0gImQiKSAlPiUgDQogIGRwbHlyOjpyZW5hbWUoZGF0ZSA9IGlfZGF0ZSkNCg0Kc2xvcGVfbGlzdFtbZF9uYW1lXV0gPC0gZF9kZg0KfQ0KDQoNCnNsb3BlX2RmIDwtIGJpbmRfcm93cyhzbG9wZV9saXN0KSAlPiUgDQogIGRwbHlyOjptdXRhdGUocmVzcF9jYXRfZGF0ZSA9IHBhc3RlMChyZXNwaXJvbWV0ZXJfZ3JvdXAsICJfIiwgc3RhcnRfZGF0ZSksDQogICAgICAgICAgICAgICAgY2hhbWJlcl9uID0gc3RyX2V4dHJhY3QoY2hhbWJlcl9pZCwgIlxcZCsiKSwNCiAgICAgICAgICAgICAgICBpZF9wcm94ID0gcGFzdGUwKHJlc3BfY2F0X2RhdGUsICJfIiwgY2hhbWJlcl9uKSwNCiAgICAgICAgICAgICAgICB0aW1lX2htcyA9IGFzX2htcyh0aW1lKjM2MDApLA0KICAgICAgICAgICAgICAgIGRhdGVfY2hyID0gZm9ybWF0KGRhdGUsICIlZC8lbS8lWSIpDQogICAgICAgICAgICAgICAgKQ0KYGBgDQoNCiMjIEZpc2ggbWV0YWRhdGENCg0KKirwn5K/IG1ldGFkYXRhKio6IFRoaXMgaXMgdGhlIG1ldGEgZGF0YSBmb3IgZWFjaCBjaGFtYmVyDQoNCipOb3RlOiBXZSBhcmUgYWxzbyBhZGRpbmcgdm9sdW1lIGJhc2VkIG9uIGNoYW1iZXIgdHlwZS4qDQoNCmBgYHtyfQ0Kc2V0d2QoaW5wdXRfZGF0YV93ZCkNCg0KbWV0YWRhdGEgPC0gcmVhZF9leGNlbCgiZmlzaC1tZXRhLnhsc3giLCBuYSA9ICJOQSIpICU+JSANCiAgZHBseXI6Om11dGF0ZShpZF9zcGxpdCA9IGlkKSAlPiUgDQogIHRpZHlyOjpzZXBhcmF0ZShpZF9zcGxpdCwgaW50byA9IGMoInJlc3Bpcm9tZXRlcl9ncm91cCIsICJzYWxpbml0eV9ncm91cCIsICJzdGFydF9kYXRlIiwgImNoYW1iZXIiKSwgc2VwID0gIl8iKSAlPiUgDQogIGRwbHlyOjptdXRhdGUoDQogICAgICB2b2x1bWUgPSBkcGx5cjo6Y2FzZV93aGVuKA0KICAgICAgICBjaGFtYmVyX3R5cGUgPT0gIkwiIH4gMC4zMDAsDQogICAgICAgIGNoYW1iZXJfdHlwZSA9PSAiTV9NIiB+IDAuMTA1LA0KICAgICAgICBjaGFtYmVyX3R5cGUgPT0gIk1fTk0iIH4gMC4xMSwNCiAgICAgICAgY2hhbWJlcl90eXBlID09ICJTIiB+IDAuMDU4LA0KICAgICAgICBjaGFtYmVyX3R5cGUgPT0gIlNNIiB+IDAuMDc1LA0KICAgICAgICBjaGFtYmVyX3R5cGUgPT0gIkQzIiB+IDAuMDU1LA0KICAgICAgICBUUlVFIH4gTkENCiAgICAgICksDQogICAgICBpZF9wcm94ID0gcGFzdGUwKHJlc3Bpcm9tZXRlcl9ncm91cCwgIl8iLCBzdGFydF9kYXRlLCAiXyIsIGNoYW1iZXIpLA0KICAgICAgcmVsX3NpemUgPSBtYXNzL3ZvbHVtZSkNCmBgYA0KDQojIyBVcmJpbmEsIEdsb3ZlciwgYW5kIEZvcnN0ZXIgKDIwMTIpDQoNCioq8J+SvyB1cmJpbmFfZXRfYWxfMjAxMioqOiBUaGlzIGlzIHRoZSBtZWFuIGxldmVsIGRhdGEgZXh0cmFjdGVkIGZyb20gVXJiaW5hLCBHbG92ZXIsIGFuZCBGb3JzdGVyICgyMDEyKV5bMl1eIEZpZ3VyZSAxYS4gV2UgdXNlZCB0aGUgbWV0YURpZ2l0aXNlIHBhY2thZ2UgaW4gUiB0byBleHRyYWN0IHRoZSBkYXRhIF5bM11eLg0KDQpgYGB7cn0NCnNldHdkKGlucHV0X2RhdGFfd2QpDQoNCnVyYmluYV9ldF9hbF8yMDEyIDwtIHJlYWRfZXhjZWwoInVyYmluYS1ldC1hbC0yMDEyLWZpZzFhLnhsc3giKQ0KYGBgDQoNCiMjIE/igoJjcml0IHZpdXNhbCBpbnNwZWN0aW9uDQoNCioq8J+SvyBvMmNyaXRfY2hlY2sqKjogVGhpcyBkYXRhIGZyYW1lIGluY2x1ZGVzIHRoZSB2aXN1YWwgYmFzZWQgYXNzZXNzbWVudCBmb3IgT+KCgmNyaXQgZm9yIGFsbCBmaXNoIGluY2x1ZGVkIGluIHRoZSBhbmFseXNpcyAoKm4qID0gNTgpLiBUaGUgaW5zcGVjdGlvbnMgd2VyZSBtYWRlIHVzaW5nIGZpZ3VyZXMgZ2VuYnJhdGVkIGJ5IHRoZSBgcGxvdE8yY3JpdCgpYCBmdW5jdGlvbiwgYW5kIGNhbiBiZSB2aWV3ZWQgaW4gYGNvbWJpbmVkX2NoYWJvdF9wbG90cy5wZGZgLiBUaGUgdmlzdWFsIGFzc2Vzc21lbnQgd2FzIGRvbmUgaW5kZXBlbmRlbnRseSBieSBhbGwgYXV0aG9ycywgYW5kIHRoZSBwcmVzZW5jZSBvZiBhbiBP4oKCY3JpdHMgYWxsb2NhdGVkIHRvICd5ZXMnLCAnbm8nLCAnbWF5YmUnLiBJZiB5ZXMsIG9yIG1heWJlLCBhbiBlc3RpbWF0ZSBvZiB0aGUgZGlzc29sdmVkIG94eWdlbiBwZXJjZW50YWdlIHdhcyBnaXZlbi4gDQoNCkRpcmVjdG9yeSB0byB0aGUgZmlndXJlcyAgIA0KDQrwn5OBIGdtYWMtbW8yLTI0Lw0K4pSU4pSA4pSAIPCfk4Igb3V0cHV0LWZpZy8gIA0KICAgIOKUlOKUgOKUgCDwn5OCIG1vZGVsX2NoYWJvdC8gDQogICAgICAgICDilJTilIDilIAg8J+ThCBjb21iaW5lZF9jaGFib3RfcGxvdHMucGRmICAgDQoNCmBgYHtyfQ0Kc2V0d2QoaW5wdXRfZGF0YV93ZCkNCg0KbzJjcml0X2NoZWNrIDwtIHJlYWRfZXhjZWwoIm8yY3JpdC12aXN1YWwtYXNzZXNzbWVudC54bHN4IiwgbmEgPSAiTkEiKQ0KYGBgDQoNCiMjIyBDb21iaW5kaW5nIG1ldGFkYXRhDQoNCkFkZGluZyB0aGUgbWV0YSBkYXRhIHRvIHRoZSBzbG9wZXMgZGF0YSBmcmFtZQ0KDQpgYGB7cn0NCnNsb3BlX2RmXzIgPC0gc2xvcGVfZGYgJT4lIA0KICBkcGx5cjo6c2VsZWN0KC1zdGFydF9kYXRlLCAtcmVzcGlyb21ldGVyX2dyb3VwKSAlPiUgDQogIGxlZnRfam9pbihtZXRhZGF0YSwgYnkgPSAiaWRfcHJveCIpICU+JSANCiAgZHBseXI6Om11dGF0ZShsaWdodF9kYXJrID0gaWZfZWxzZSh0aW1lX2htcyA+PSBhcy5obXMoIjA3OjAwOjAwIikgJiB0aW1lX2htcyA8IGFzLmhtcygiMTk6MDA6MDAiKSwgImxpZ2h0IiwgImRhcmsiKSkgJT4lIA0KICBkcGx5cjo6YXJyYW5nZShpZCkNCmBgYA0KDQo8IS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0+DQoNCiMg8J+nuSBEYXRhIHRpZHkNCg0KPCEtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tPg0KDQojIyBOdW1iZXJzDQoNCldlIGhhdmUgKio2NCBmaXNoKiogd2l0aCBNT+KCgiBkYXRhDQoNCmBgYHtyfQ0KbiA8LSBzbG9wZV9kZl8yICU+JSANCiAgZHBseXI6OmZpbHRlcihjaGFtYmVyX2NvbmRpdGlvbiA9PSAiZmlzaCIpICU+JSANCiAgZHBseXI6OmRpc3RpbmN0KGlkKSAlPiUgDQogIG5yb3coLikNCg0KcGFzdGUwKCJuID0gIiwgbikNCmBgYA0KDQpXaXRoIDQ4IGZyb20gdGhlIDAgcHB0IGFuZCA0OCBmcm9tIDkgcHB0IGdyb3Vwcw0KDQpgYGB7cn0NCnNsb3BlX2RmXzIgJT4lIA0KICBkcGx5cjo6ZmlsdGVyKGNoYW1iZXJfY29uZGl0aW9uID09ICJmaXNoIikgJT4lDQogIGRwbHlyOjpncm91cF9ieShzYWxpbml0eV9ncm91cCkgJT4lIA0KICBkcGx5cjo6cmVmcmFtZSgnbiB0b3RhbCcgPSBsZW5ndGgodW5pcXVlKGlkKSkpICU+JSANCiAgZ3QoKSAlPiUgDQogIGNvbHNfbGFiZWwoDQogICAgc2FsaW5pdHlfZ3JvdXAgPSAiU2FsaW5pdHkgZ3JvdXAiDQogICkgJT4lIA0KICBjb2xzX2FsaWduKA0KICAgIGFsaWduID0gImNlbnRlciIsIA0KICAgIGNvbHVtbnMgPSBldmVyeXRoaW5nKCkNCiAgKQ0KYGBgDQoNCiMjIEZpc2ggc2l6ZQ0KDQpIZXJlIHdlIGNhbGN1bGF0ZSB0aGUgbWVhbiBsZW5ndGggYW5kIHNpemUgb2YgZmlzaCB1c2VkIGluIHRoZSBleHBlcmltZW50Lg0KDQpgYGB7cn0NCm1hc3NfbGVuZ3RoIDwtIHNsb3BlX2RmXzIgJT4lIA0KICBkcGx5cjo6ZmlsdGVyKGNoYW1iZXJfY29uZGl0aW9uID09ICJmaXNoIikgJT4lDQogIGRwbHlyOjpncm91cF9ieShpZCkgJT4lIA0KICBkcGx5cjo6c2FtcGxlX24oMSkgJT4lIA0KICBkcGx5cjo6dW5ncm91cCgpICU+JSANCiAgZHBseXI6OnJlZnJhbWUoeF9tYXNzID0gcm91bmQobWVhbihtYXNzLCBuYS5ybSA9IFRSVUUpLCAzKSwNCiAgICAgICAgICAgICAgICAgbWluX21hc3MgPSByb3VuZChtaW4obWFzcywgbmEucm0gPSBUUlVFKSwgMyksDQogICAgICAgICAgICAgICAgIG1heF9tYXNzID0gcm91bmQobWF4KG1hc3MsIG5hLnJtID0gVFJVRSksIDMpLA0KICAgICAgICAgICAgICAgICB4X2xlbmd0aCA9IHJvdW5kKG1lYW4obGVuZ3RoLCBuYS5ybSA9IFRSVUUpLCAyKSwNCiAgICAgICAgICAgICAgICAgbWluX2xlbmd0aCA9IHJvdW5kKG1pbihsZW5ndGgsIG5hLnJtID0gVFJVRSksIDIpLA0KICAgICAgICAgICAgICAgICBtYXhfbGVuZ3RoID0gcm91bmQobWF4KGxlbmd0aCwgbmEucm0gPSBUUlVFKSwgMikpDQoNCm1hc3NfbWVhbiA8LSBtYXNzX2xlbmd0aCAlPiUgDQogIHB1bGwoeF9tYXNzKQ0KDQptYXNzX21pbiA8LSBtYXNzX2xlbmd0aCAlPiUgDQogIHB1bGwobWluX21hc3MpDQoNCm1hc3NfbWF4IDwtIG1hc3NfbGVuZ3RoICU+JSANCiAgcHVsbChtYXhfbWFzcykNCg0KbGVuZ3RoX21lYW4gPC0gbWFzc19sZW5ndGggJT4lIA0KICBwdWxsKHhfbGVuZ3RoKQ0KDQpsZW5ndGhfbWluIDwtIG1hc3NfbGVuZ3RoICU+JSANCiAgcHVsbChtaW5fbGVuZ3RoKQ0KDQpsZW5ndGhfbWF4IDwtIG1hc3NfbGVuZ3RoICU+JSANCiAgcHVsbChtYXhfbGVuZ3RoKQ0KDQpwYXN0ZTAoIlRoZSBtZWFuIG1hc3Mgb2YgZmlzaCB3YXMgIiwgbWFzc19tZWFuLCAiIGcgKHJhbmdlOiAiLCBtYXNzX21pbiwgIi0iLCBtYXNzX21heCwgIikiLA0KICAgICAgICIsIGFuZCB0aGUgbWVhbiBsZW5ndGggd2FzICIsIGxlbmd0aF9tZWFuLCAiIG1tIChyYW5nZTogIiwgbGVuZ3RoX21pbiwgIi0iLCBsZW5ndGhfbWF4LCAiKSIpDQpgYGANCg0KIyMgRmlsdGVyaW5nIHRyaWFscw0KDQpXZSB3aWxsIHJlbW92ZSA2IHRyaWFscyB3aGljaCBoYWQgZXJyb3JzLiBUaGVzZSBhcmUgYXMgZm9sbG93czoNCg0KLSAgIGFfMF8yNW5vdl8zIGZpc2ggZGllZCBkdXJpbmcgdHJpYWwNCi0gICBiXzBfMjZub3ZfNCBmbGF0IGxpbmVkIGVhcmx5DQotICAgY18wXzIybm92XzIgY2hhbWJlciB3YXMgb3BlbmVkIGVhcmx5DQotICAgY185XzI2bm92XzIgY2hhbWJlciB3YXMgb3BlbmVkIGVhcmx5DQotICAgY185XzI2bm92XzQgY2hhbWJlciB3YXMgb3BlbmVkIGVhcmx5DQotICAgZF85XzI3bm92XzMgc2Vuc29yIHdhcyBqdW1weSBhbmQgZW5kIHBvaW50cyB3ZXJlIGhhcmQgdG8gY29uZmlkZW50bHkgSUQgdmlzdWFsbHkNCg0KYGBge3J9DQpyZW1vdmVfdHJpYWxfZXJyb3IgPC0gYygiYV8wXzI1bm92XzMiLCAiYl8wXzI2bm92XzQiLCAiY18wXzIybm92XzIiLCAiY185XzI2bm92XzIiLCAiY185XzI2bm92XzQiLCAiZF85XzI3bm92XzMiKQ0KDQpzbG9wZV9kZl9maWx0ZXIgPC0gc2xvcGVfZGZfMiAlPiUgDQogIGRwbHlyOjpmaWx0ZXIoIShpZCAlaW4lIHJlbW92ZV90cmlhbF9lcnJvcikpDQpgYGANCg0KV2Ugbm93IGhhdmUgKio1OCBmaXNoKiogd2l0aCBNTzIgZGF0YQ0KDQpgYGB7cn0NCm4gPC0gc2xvcGVfZGZfZmlsdGVyICU+JSANCiAgZHBseXI6OmZpbHRlcihjaGFtYmVyX2NvbmRpdGlvbiA9PSAiZmlzaCIpICU+JSANCiAgZHBseXI6OmRpc3RpbmN0KGlkKSAlPiUgDQogIG5yb3coLikNCg0KcGFzdGUwKCJuID0gIiwgbikNCmBgYA0KDQpXaXRoIDMwIGluIHRoZSAwIHBwdCBncm91cCBhbmQgMjggaW4gdGhlIDkgcHB0IGdyb3VwDQoNCmBgYHtyfQ0Kc2xvcGVfZGZfZmlsdGVyICU+JSANCiAgZHBseXI6OmZpbHRlcihjaGFtYmVyX2NvbmRpdGlvbiA9PSAiZmlzaCIpICU+JQ0KICBkcGx5cjo6Z3JvdXBfYnkoc2FsaW5pdHlfZ3JvdXApICU+JSANCiAgZHBseXI6OnJlZnJhbWUoJ24gdG90YWwnID0gbGVuZ3RoKHVuaXF1ZShpZCkpKSAlPiUgDQogIGd0KCkgJT4lIA0KICBjb2xzX2xhYmVsKA0KICAgIHNhbGluaXR5X2dyb3VwID0gIlNhbGluaXR5IGdyb3VwIg0KICApICU+JSANCiAgY29sc19hbGlnbigNCiAgICBhbGlnbiA9ICJjZW50ZXIiLCANCiAgICBjb2x1bW5zID0gZXZlcnl0aGluZygpDQogICkNCmBgYA0KDQpDaGFtYmVyIHNpemUgYnJlYWtkb3duDQoNCmBgYHtyfQ0Kc2xvcGVfZGZfZmlsdGVyICU+JSANCiAgZHBseXI6OmZpbHRlcihjaGFtYmVyX2NvbmRpdGlvbiA9PSAiZmlzaCIpICU+JQ0KICBkcGx5cjo6bXV0YXRlKHZvbHVtZSA9IGlmX2Vsc2Uodm9sdW1lID09IDAuMDU1LCAwLjA1OCwgdm9sdW1lKSkgJT4lIA0KICBkcGx5cjo6Z3JvdXBfYnkodm9sdW1lKSAlPiUgDQogIGRwbHlyOjpyZWZyYW1lKCduIHRvdGFsJyA9IGxlbmd0aCh1bmlxdWUoaWQpKSwNCiAgICAgICAgICAgICAgICAgbWVkaWFuX21hc3MgPSBtZWRpYW4obWFzcywgbmEucm0gPSBUUlVFKSwNCiAgICAgICAgICAgICAgICAgc2RfbWFzcyA9IHNkKG1hc3MsIG5hLnJtID0gVFJVRSkpICU+JSANCiAgZ3QoKSAlPiUgDQogIGNvbHNfbGFiZWwoDQogICAgdm9sdW1lID0gIkNoYW1iZXIgdHlwZSIsDQogICAgbWVkaWFuX21hc3MgPSAiTWVkaWFuIG1hc3MiLA0KICAgIHNkX21hc3MgPSAiU3RhbmRhcmQgZGV2aWF0aW9uIG1hc3MiDQogICkgJT4lIA0KICBjb2xzX2FsaWduKA0KICAgIGFsaWduID0gImNlbnRlciIsIA0KICAgIGNvbHVtbnMgPSBldmVyeXRoaW5nKCkNCiAgKQ0KYGBgDQoNCk51bWJlcnMgcGVyIHRyaWFsIGFwcHJvYWNoDQoNCmBgYHtyfQ0KcGhhc2VzIDwtIGMoIjUwYyIsICIxMDBjIikNCg0Kc2xvcGVfZGZfZmlsdGVyICU+JSANCiAgZHBseXI6OmZpbHRlcihjaGFtYmVyX2NvbmRpdGlvbiA9PSAiZmlzaCIgJiBwaGFzZSAlaW4lIHBoYXNlcykgJT4lDQogIGRwbHlyOjpncm91cF9ieShwaGFzZSkgJT4lIA0KICBkcGx5cjo6cmVmcmFtZSgnbiB0b3RhbCcgPSBsZW5ndGgodW5pcXVlKGlkKSkpICU+JSANCiAgZ3QoKSAlPiUgDQogIGNvbHNfbGFiZWwoDQogICAgcGhhc2UgPSAiVHlwZSBvZiB0cmlhbCIsDQogICkgJT4lIA0KICBjb2xzX2FsaWduKA0KICAgIGFsaWduID0gImNlbnRlciIsIA0KICAgIGNvbHVtbnMgPSBldmVyeXRoaW5nKCkNCiAgKQ0KYGBgDQoNCiMjIEZpbHRlcmluZyBNT+KCgiBlc3RpbWF0ZXMNCg0KSGVyZSB3ZSBhcHBseSB0aGUgZm9sbG93aW5nIGZpbHRlcnMgdG8gdGhlIE1P4oKCIGRhdGE6DQoNCi0gICBSZW1vdmUgdGhlIGZpcnN0IDUgU01SIGN5Y2xlcyAoYnVybiBpbikNCi0gICBSZW1vdmUgYWxsIHBvc2l0aXZlIHJhdyBzbG9wZXMNCi0gICBSZW1vdmUgYWxsIE1P4oKCIGNhbGN1bGF0ZWQgdXNpbmcgbGVzcyB0aGVuIDYwIGRhdGEgcG9pbnRzICg1IG1pbiBhdCAwLjIgSHopDQotICAgUmVtb3ZlIGFsbCBNT+KCgiBjYWxjdWxhdGVkIGlmIE/igoIgaW5jcmVhc2VzIGluIGEgY2xvc2VkIHBoYXNlIChpLmUuIHRyaWFsIGhhcyBlbmRlZCkNCg0KQ2hlY2sgcG9zaXRpdmUgdmFsdWVzIGZvciBNT+KCgiBiZWZvcmUgcmVtb3ZpbmcuDQoNCmBgYHtyfQ0Kc2xvcGVfdGlkeV9yZW1vdmVfZmx1c2ggPC0gc2xvcGVfZGZfZmlsdGVyICU+JQ0KICBkcGx5cjo6ZmlsdGVyKHBoYXNlICE9ICJzbXIiLCBjaGFtYmVyX2NvbmRpdGlvbiA9PSAiZmlzaCIpICU+JSANCiAgZHBseXI6Omdyb3VwX2J5KGlkKSAlPiUNCiAgZHBseXI6OmFycmFuZ2UoaWQsIG9yZGVyKSAlPiUgICMgRW5zdXJlIHRoZSBkYXRhIGlzIHNvcnRlZCB3aXRoaW4gZWFjaCBncm91cA0KICBkcGx5cjo6bXV0YXRlKG8yX2RpZmYgPSBpZl9lbHNlKHJvd19udW1iZXIoKSA9PSAxLCAwLCBvMiAtIGxhZyhvMikpLCAjIENhbGN1bGF0ZSB0aGUgZGlmZmVyZW5jZSBpbiAnbzInDQogICAgICAgICAgICAgICAgbzJfZGlmZl9jdW1zdW0gPSBjdW1zdW0obzJfZGlmZiA+IDEpKSAlPiUgICMgQ2hlY2tzIGZpcnN0IG9jY3VycmVuY2UgYW5kIHN1bXMgDQogIGRwbHlyOjpmaWx0ZXIobzJfZGlmZl9jdW1zdW0gPT0gMCkgJT4lICAjIEtlZXAgcm93cyB1bnRpbCB0aGUgZmlyc3QganVtcCA+IDENCiAgZHBseXI6OnVuZ3JvdXAoKSAlPiUNCiAgZHBseXI6OnNlbGVjdCgtbzJfZGlmZiwgLW8yX2RpZmZfY3Vtc3VtKQ0KDQpwb3N0aXZlX3Nsb3BlcyA8LSBzbG9wZV90aWR5X3JlbW92ZV9mbHVzaCAlPiUNCiAgZHBseXI6OmZpbHRlcihjaGFtYmVyX2NvbmRpdGlvbiA9PSAiZmlzaCIgJiBtbzJjb3JyID4gMCkNCg0Kbl9wb3N0aXZlX3Nsb3BlcyA8LSBucm93KHBvc3RpdmVfc2xvcGVzKQ0KDQpuX3Nsb3BlcyA8LSAgbnJvdyhzbG9wZV90aWR5X3JlbW92ZV9mbHVzaCkNCg0KbGlzdF9wb3N0aXZlX2FsbCA8LSBwb3N0aXZlX3Nsb3BlcyAlPiUgDQogIGRwbHlyOjpkaXN0aW5jdChpZCkgJT4lIA0KICBkcGx5cjo6cHVsbChpZCkNCg0KcHJpbnQocGFzdGUwKCJUaGVyZSBhcmUgIiwgbGVuZ3RoKGxpc3RfcG9zdGl2ZV9hbGwpLCAiIGZpc2ggd2l0aCBwb3N0aXZlIHNsb3Blcy4gVGhlc2UgZmlzaCBhcmU6ICIsIHBhc3RlKGxpc3RfcG9zdGl2ZV9hbGwsIGNvbGxhcHNlID0gIiwgIiksICIuIEZvciBhbGwgZXN0aW1hdGVkIHNsb3BlcyAobiA9ICIsIG5fc2xvcGVzLCAiKSAiLCByb3VuZCgobl9wb3N0aXZlX3Nsb3Blcy9uX3Nsb3BlcykqMTAwLDIpICwgIiUgd2VyZSBwb3N0aXZlIChuID0gIiwgbl9wb3N0aXZlX3Nsb3BlcywgIikiKSkNCmBgYA0KDQpGaWx0ZXJpbmcgdGhlIE1P4oKCIGRhdGENCg0KYGBge3J9DQpjeWNsZV9idXJuIDwtIDA6NA0KDQpzbG9wZV9kZl9maWx0ZXJfMSA8LSBzbG9wZV9kZl9maWx0ZXIgJT4lDQogIGRwbHlyOjpmaWx0ZXIoIShjeWNsZSAlaW4lIGN5Y2xlX2J1cm4pICYgDQogICAgICAgICAgICAgICAgICBtbzJjb3JyIDwgMCAmIA0KICAgICAgICAgICAgICAgICAgbiA+IDYwICYNCiAgICAgICAgICAgICAgICAgIGNoYW1iZXJfY29uZGl0aW9uID09ICJmaXNoIg0KICAgICAgICAgICAgICAgICkNCiAgDQojIEZpbHRlciBvdXQgdGhlIGVuZCBmbHVzaA0Kc2xvcGVfdGlkeV9jbG9zZWQgPC0gc2xvcGVfZGZfZmlsdGVyXzEgJT4lDQogIGRwbHlyOjpmaWx0ZXIocGhhc2UgIT0gInNtciIpICU+JSANCiAgZHBseXI6Omdyb3VwX2J5KGlkKSAlPiUNCiAgZHBseXI6OmFycmFuZ2UoaWQsIG9yZGVyKSAlPiUgICMgRW5zdXJlIHRoZSBkYXRhIGlzIHNvcnRlZCB3aXRoaW4gZWFjaCBncm91cA0KICBkcGx5cjo6bXV0YXRlKG8yX2RpZmYgPSBpZl9lbHNlKHJvd19udW1iZXIoKSA9PSAxLCAwLCBvMiAtIGxhZyhvMikpLCAjIENhbGN1bGF0ZSB0aGUgZGlmZmVyZW5jZSBpbiAnbzInDQogICAgICAgICAgICAgICAgbzJfZGlmZl9jdW1zdW0gPSBjdW1zdW0obzJfZGlmZiA+IDEpKSAlPiUgICMgQ2hlY2tzIGZpcnN0IG9jY3VycmVuY2UgYW5kIHN1bXMgDQogIGRwbHlyOjpmaWx0ZXIobzJfZGlmZl9jdW1zdW0gPT0gMCkgJT4lICAjIEtlZXAgcm93cyB1bnRpbCB0aGUgZmlyc3QganVtcCA+IDENCiAgZHBseXI6OnVuZ3JvdXAoKSAlPiUNCiAgZHBseXI6OnNlbGVjdCgtbzJfZGlmZiwgLW8yX2RpZmZfY3Vtc3VtKQ0KDQpzbG9wZV90aWR5X3NtciA8LSBzbG9wZV9kZl9maWx0ZXJfMSAlPiUgDQogIGRwbHlyOjpmaWx0ZXIocGhhc2UgPT0gInNtciIpDQoNCnNsb3BlX2RmX2ZpbHRlcl8yIDwtIHJiaW5kKHNsb3BlX3RpZHlfc21yLCBzbG9wZV90aWR5X2Nsb3NlZCkgJT4lIA0KICBkcGx5cjo6YXJyYW5nZShpZCwgb3JkZXIpDQpgYGANCg0KIyMgQ2FsY3VsYXRpbmcgU01SDQoNCldlIHdpbGwgY2FsY3VsYXRlIFNNUiB1c2luZyBgY2FsY1NNUmAgZnVuY3Rpb24gYnkgQ2hhYm90LCBTdGVmZmVuc2VuIGFuZCBGYXJyZWxsICgyMDE2KV5bMV1eLiBTcGVjaWZpY2FsbHksIHdlIHVzZSBtZWFuIG9mIHRoZSBsb3dlc3Qgbm9ybWFsIGRpc3RyaWJ1dGlvbiAoTUxORCkgd2hlcmUgQ1ZtbG5kIFw8IDUuNCBhbmQgdGhlIG1lYW4gb2YgdGhlIGxvd2VyIDIwJSBxdWFudGlsZSAocTAuMikgd2VyZSBDVm1sbmQgXD4gNS40LiBJZiBDVm1sbmQgaXMgbm90IGNhbGN1bGF0ZWQgd2UgaGF2ZSB1c2VkIHEwLjIuDQoNCmBgYHtyfQ0KbGFiY2hhcnRfY2hhYm90X3NtciA8LSBzbG9wZV9kZl9maWx0ZXJfMiAlPiUNCiAgZHBseXI6OmZpbHRlcihwaGFzZSA9PSAic21yIikNCg0KIyBFeHRyYWN0IGRpc3RpbmN0IElEcw0KaWRzIDwtIGxhYmNoYXJ0X2NoYWJvdF9zbXIgJT4lIA0KICBkcGx5cjo6ZGlzdGluY3QoaWQpICU+JSANCiAgZHBseXI6OnB1bGwoKQ0KDQojIEluaXRpYWxpc2UgYW4gZW1wdHkgbGlzdCB0byBzdG9yZSBTTVIgZGF0YQ0Kc21yX2xpc3QgPC0gbGlzdCgpDQoNCiMgUHJvY2VzcyBlYWNoIElEDQpmb3IgKGlkX2kgaW4gaWRzKSB7DQogIHRyeUNhdGNoKHsNCiAgICAjIEZpbHRlciB0aGUgZGF0YSBmb3IgdGhlIGN1cnJlbnQgSUQNCiAgICBkZl9pIDwtIGxhYmNoYXJ0X2NoYWJvdF9zbXIgJT4lIA0KICAgICAgZHBseXI6OmZpbHRlcihpZCA9PSBpZF9pKSAlPiUgDQogICAgICBkcGx5cjo6bXV0YXRlKGFic19tbzJjb3JyID0gYWJzKG1vMmNvcnIpKQ0KICAgIA0KICAgICMgQ2FsY3VsYXRlIFNNUiByZXN1bHRzDQogICAgY2FsY1NNUl9yZXN1bHRzIDwtIGNhbGNTTVIoZGZfaSRhYnNfbW8yY29ycikNCiAgICBDVm1sbmRfaSA8LSBjYWxjU01SX3Jlc3VsdHMkQ1ZtbG5kDQogICAgcXVhbnRfaSA8LSBjYWxjU01SX3Jlc3VsdHMkcXVhbnQgJT4lIGFzX3RpYmJsZSgpDQogICAgcXVhbnRfMjBwZXJfaSA8LSBxdWFudF9pJHZhbHVlWzNdDQogICAgbWxuZF9pIDwtIGNhbGNTTVJfcmVzdWx0cyRtbG5kDQogICAgc21yX3ZhbHVlIDwtIGlmX2Vsc2UoQ1ZtbG5kX2kgPCA1LjQsIG1sbmRfaSwgcXVhbnRfMjBwZXJfaSkNCiAgICBzbXJfdHlwZSA8LSBpZl9lbHNlKENWbWxuZF9pIDwgNS40LCAibWxuZCIsICJxdWFudF8yMHBlciIpDQogICAgc21yX3ZhbHVlIDwtIGlmX2Vsc2UoaXMubmEoc21yX3ZhbHVlKSwgcXVhbnRfMjBwZXJfaSwgc21yX3ZhbHVlKQ0KICAgIHNtcl90eXBlIDwtIGlmX2Vsc2UoaXMubmEoc21yX3R5cGUpLCAicXVhbnRfMjBwZXIiLCBzbXJfdHlwZSkNCiAgICANCiAgICAjIENyZWF0ZSBhIGRhdGEgZnJhbWUgZm9yIHRoZSBjdXJyZW50IElEDQogICAgc21yX2RmX2kgPC0gdGliYmxlOjp0aWJibGUoDQogICAgICBpZCA9IGlkX2ksDQogICAgICBzbXIgPSBzbXJfdmFsdWUsDQogICAgICBzbXJfZXN0ID0gc21yX3R5cGUNCiAgICApDQogICAgDQogIH0sIGVycm9yID0gZnVuY3Rpb24oZSkgew0KICAgICMgSGFuZGxlIGVycm9ycyBieSBhc3NpZ25pbmcgTkEgdmFsdWVzDQogICAgc21yX2RmX2kgPC0gdGliYmxlOjp0aWJibGUoDQogICAgICBpZCA9IGlkX2ksDQogICAgICBzbXIgPSBOQSwNCiAgICAgIHNtcl9lc3QgPSBOQQ0KICAgICkNCiAgfSkNCiAgDQogICMgQXBwZW5kIHRvIHRoZSBsaXN0DQogIHNtcl9saXN0W1tpZF9pXV0gPC0gc21yX2RmX2kNCn0NCg0KIyBDb21iaW5lIGFsbCBpbmRpdmlkdWFsIFNNUiBkYXRhIGZyYW1lcyBpbnRvIG9uZQ0Kc21yX2RmIDwtIGJpbmRfcm93cyhzbXJfbGlzdCkgJT4lIA0KICBkcGx5cjo6cmVuYW1lKHNtcl9jaGFib3QgPSBzbXIsDQogICAgICAgICAgICAgICAgc21yX2NoYWJvdF9tZXRob2QgPSBzbXJfZXN0KQ0KDQpzbG9wZV9kZl9maWx0ZXJfMyA8LSBzbG9wZV9kZl9maWx0ZXJfMiAlPiUNCiAgZHBseXI6OmxlZnRfam9pbiguLCBzbXJfZGYsIGJ5ID0gImlkIikNCmBgYA0KDQojIyBUcmFuc2Zvcm1pbmcgTU/igoINCg0KSGVyZSB3ZSBhcmUgdHJhbnNmb3JtaW5nIHRoZSBNT+KCgiB1bml0cy4gVGhlIHJlc3VsdGluZyB2YWx1ZXMgYXJlIGFzIGZvbGxvd3M6DQoNCi0gICAqKk1PMioqIGlzIGFic29sdXRlIHZhbHVlIG9mIHRoZSBiYWNrZ3JvdW5kIGFuZCBsZWFrIGNvcnJlY3RlZCBNT+KCgiBzbG9wZSBmcm9tIExhYmNoYXJ0IChtbzJjb3JyKSB0aW1lcyB0aGUgbmV0IHZvbHVtZSBvZiB0aGUgY2hhbWJlciAodm9sdW1lIC0gZmlzaCBtYXNzKSwgw5cgNjAsIMOXIDYwLCB0byBhY2hpZXZlIE1P4oKCIGFzIG1nXi0xXiBP4oKCIGheLTFeDQoNCi0gICAqKk1PMl9nKiogaXMgTU8yIGRpdmlkZWQgYnkgZmlzaCBtYXNzIHRvIGFjaGlldmUgTU/igoIgYXMgbWdeLTFeIE/igoIgZ14tMV4gaF4tMV4gKGkuZS4gbWFzcyBzdGFuZGFyZGlzZWQpDQoNCi0gICAqKlNNUioqIGFic29sdXRlIHZhbHVlIG9mIHRoZSBTTVIgZXN0aW1hdGVzIHVzaW5nIG1ldGhvZHMgZGVzY3JpYmVkIGJ5IENoYWJvdCwgU3RlZmZlbnNlbiBhbmQgRmFycmVsbCAoMjAxNileWzFdXiB0aW1lcyB0aGUgbmV0IHZvbHVtZSBvZiB0aGUgY2hhbWJlciAodm9sdW1lIC0gZmlzaCBtYXNzKSwgw5cgNjAsIMOXIDYwLCB0byBhY2hpZXZlIFNNUiBhcyBtZ14tMV4gT+KCgiBoXi0xXikNCg0KLSAgICoqU01SX2cqKiBpcyBTTVIgZGl2aWRlZCBieSBmaXNoIG1hc3MgdG8gYWNoaWV2ZSBTTVIgYXMgbWdeLTFeIE/igoIgZ14tMV4gaF4tMV4gKGkuZS4gbWFzcyBzdGFuZGFyZGlzZWQpDQoNCi0gICAqKkRPKiogaXMgZGlzc29sdmVkIG94eWdlbiBwZXJjZW50YWdlIGNhbGN1bGF0ZWQgZnJvbSBP4oKCIHZhbHVlcyAobWdeLTFeIExeLTFeKSB1c2luZyB0aGUgcmVjb3JkZWQgdGVtcGVyYXR1cmUsIHNhbGluaXR5LCBhbmQgYSBjb25zdGFudCBhdG1vc3BoZXJpYyBwcmVzc3VyZSAoUGE7IDEwMTMuMjUpDQoNCi0gICAqKm8yX2twYSoqIGlzIHRoZSBP4oKCIGNvbmNlbnRyYXRpb24gaW4ga2lsb3Bhc2NhbCAoa3BhKS4gVGhpcyBpcyB1c2VkIHRvIG1ha2UgYSBjb21wYXJhdGl2ZSBmaWd1cmUgb25seS4NCg0KYGBge3J9DQojIENvbWJpbmUgYmFjayBpbnRvIG9uZSBkYXRhIGZyYW1lDQpzbG9wZV90aWR5IDwtIHNsb3BlX2RmX2ZpbHRlcl8zICU+JSANCiAgICBkcGx5cjo6bXV0YXRlKERPID0gY29udl9vMigNCiAgICAgICAgICAgICAgICAgICBvMiA9IG8yLA0KICAgICAgICAgICAgICAgICAgIGZyb20gPSAibWdfcGVyX2wiLA0KICAgICAgICAgICAgICAgICAgIHRvID0gInBlcmNlbnRfYS5zLiIsDQogICAgICAgICAgICAgICAgICAgdGVtcCA9IHRlbXAsICNDDQogICAgICAgICAgICAgICAgICAgc2FsID0gbWVhc3VyZWRfc2FsaW5pdHksDQogICAgICAgICAgICAgICAgICAgYXRtX3ByZXMgPSAxMDEzLjI1KSwNCiAgICAgICAgICAgICAgICAgIG8yX2twYSA9IGNvbnZfbzIoDQogICAgICAgICAgICAgICAgICAgbzIgPSBvMiwNCiAgICAgICAgICAgICAgICAgICBmcm9tID0gIm1nX3Blcl9sIiwNCiAgICAgICAgICAgICAgICAgICB0byA9ICJrUGEiLA0KICAgICAgICAgICAgICAgICAgIHRlbXAgPSB0ZW1wLCAjQw0KICAgICAgICAgICAgICAgICAgIHNhbCA9IG1lYXN1cmVkX3NhbGluaXR5LA0KICAgICAgICAgICAgICAgICAgIGF0bV9wcmVzID0gMTAxMy4yNSksDQogICAgICAgICAgICAgICAgICBuZXRfdm9sdW1lID0gdm9sdW1lIC0gbWFzcy8xMDAwLA0KICAgICAgICAgICAgICAgICAgTU8yID0gYWJzKG1vMmNvcnIpKm5ldF92b2x1bWUqNjAqNjAsDQogICAgICAgICAgICAgICAgICBNTzJfZyA9IE1PMi9tYXNzLA0KICAgICAgICAgICAgICAgICAgU01SID0gYWJzKHNtcl9jaGFib3QpKm5ldF92b2x1bWUqNjAqNjAsDQogICAgICAgICAgICAgICAgICBTTVJfZyA9IFNNUi9tYXNzDQogICAgICAgICAgICAgICAgICApDQpgYGANCg0KVHJpYWwgbGVuZ3Rocw0KDQpgYGB7cn0NCnNsb3BlX3RpZHkgJT4lIA0KICBkcGx5cjo6ZmlsdGVyKHBoYXNlICE9ICJzbXIiKSAlPiUgDQogIGRwbHlyOjogbXV0YXRlKGRhdGV0aW1lID0gYXMuUE9TSVhjdChwYXN0ZShkYXRlLCB0aW1lX2htcyksIGZvcm1hdCA9ICIlWS0lbS0lZCAlSDolTTolUyIpKSAlPiUNCiAgZHBseXI6Omdyb3VwX2J5KGlkKSAlPiUgDQogIGRwbHlyOjpyZWZyYW1lKA0KICAgIHBoYXNlID0gcGhhc2VbMV0sDQogICAgc3RhcnRfdGltZSA9IG1pbihkYXRldGltZSwgbmEucm0gPSBUUlVFKSwNCiAgICBlbmRfdGltZSA9IG1heChkYXRldGltZSwgbmEucm0gPSBUUlVFKSwNCiAgICB0aW1lX2RpZmYgPSBkaWZmdGltZShlbmRfdGltZSwgc3RhcnRfdGltZSwgdW5pdHMgPSAiaG91cnMiKSAgIyBDaGFuZ2UgdW5pdHMgYXMgbmVlZGVkDQogICkgJT4lIA0KICBkcGx5cjo6Z3JvdXBfYnkocGhhc2UpICU+JSANCiAgZHBseXI6OnJlZnJhbWUoJ01pbmltdW0gZHVyYXRpb24nID0gcm91bmQobWluKHRpbWVfZGlmZiksMiksDQogICAgICAgICAgICAgICAgICdNYXhpbXVtIGR1cmF0aW9uJyA9IHJvdW5kKG1heCh0aW1lX2RpZmYpLDIpLA0KICAgICAgICAgICAgICAgICAnTWVkaWFuIGR1cmF0aW9uJyA9IHJvdW5kKG1lZGlhbih0aW1lX2RpZmYpLDIpKSAlPiUgDQogIGRwbHlyOjptdXRhdGUocGhhc2UgPSBpZl9lbHNlKHBoYXNlID09ICIxMDBjIiwgIkNsb3NlZCBhdCAxMDAlIGFpciBzYXR1cmF0aW9uIiwgIkNsb3NlZCBhdCA3NSUgYW5kIDUwJSBhaXIgc2F0dXJhdGlvbiIpKSAlPiUgDQogIGRwbHlyOjpyZW5hbWUoQXBwcm9hY2ggPSBwaGFzZSkgJT4lIA0KICBndCgpDQogIA0KYGBgDQoNCjwhLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLT4NCg0KIyDwn5OKIEV4cGxvcmF0b3J5IERhdGEgQW5hbHlzaXMgKEVEQSkNCg0KPCEtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tPg0KDQojIyBP4oKCIHZzIE1P4oKCDQoNCiMjIyBGaWd1cmUgUzENCg0KVGhpcyBpbnRlcmFjdGl2ZSB3YXMgdXNlZCB0byBpZGVudGlmeSBhbnkgb3V0bGllcnMsIG9yIHBvdGVudGlhbCBlcnJvcnMuDQoNCmBgYHtyfQ0KbG1fbGluZXMgPC0gc2xvcGVfdGlkeSAlPiUNCiAgZ3JvdXBfYnkoaWQpICU+JQ0KICBzdW1tYXJpc2UoDQogICAgRE9fc2VxID0gbGlzdChzZXEobWluKERPKSwgbWF4KERPKSwgbGVuZ3RoLm91dCA9IDEwMCkpLCAgIyBHZW5lcmF0ZSBhIHNlcXVlbmNlIG9mIERPIHZhbHVlcw0KICAgIE1PMl9wcmVkID0gbGlzdChwcmVkaWN0KGxtKE1PMl9nIH4gRE8sIGRhdGEgPSBjdXJfZGF0YSgpKSwgbmV3ZGF0YSA9IGRhdGEuZnJhbWUoRE8gPSBzZXEobWluKERPKSwgbWF4KERPKSwgbGVuZ3RoLm91dCA9IDEwMCkpKSkNCiAgKSAlPiUNCiAgdW5uZXN0KGMoRE9fc2VxLCBNTzJfcHJlZCkpICAjIEV4cGFuZCBsaXN0cyBpbnRvIHJvd3MNCg0KIyBDcmVhdGUgc2NhdHRlciBwbG90IHdpdGggbWFya2VycyBmb3IgZWFjaCBmaXNoDQpwIDwtIHBsb3RfbHkoDQogIGRhdGEgPSBzbG9wZV90aWR5LA0KICB4ID0gfkRPLA0KICB5ID0gfk1PMl9nLA0KICB0eXBlID0gInNjYXR0ZXIiLA0KICBtb2RlID0gIm1hcmtlcnMiLA0KICBjb2xvciA9IH5pZCwgICMgQ29sb3VyIHBvaW50cyBieSBmaXNoIElEDQogIG1hcmtlciA9IGxpc3Qob3BhY2l0eSA9IDAuNiksDQogIG5hbWUgPSB+aWQNCikNCg0KIyBBZGQgcmVncmVzc2lvbiBsaW5lcyBmb3IgZWFjaCBmaXNoDQpwIDwtIHAgJT4lDQogIGFkZF90cmFjZSgNCiAgICBkYXRhID0gbG1fbGluZXMsDQogICAgeCA9IH5ET19zZXEsDQogICAgeSA9IH5NTzJfcHJlZCwNCiAgICB0eXBlID0gInNjYXR0ZXIiLA0KICAgIG1vZGUgPSAibGluZXMiLA0KICAgIGNvbG9yID0gfmlkLCAgIyBFbnN1cmUgZWFjaCBsaW5lIG1hdGNoZXMgaXRzIGNvcnJlc3BvbmRpbmcgZmlzaA0KICAgIGxpbmUgPSBsaXN0KHdpZHRoID0gMSwgZGFzaCA9ICJzb2xpZCIpLA0KICAgIHNob3dsZWdlbmQgPSBGQUxTRSAgIyBBdm9pZCBjbHV0dGVyaW5nIHRoZSBsZWdlbmQNCiAgKQ0KDQojIEZpbmFsIGxheW91dA0KcCA8LSBwICU+JQ0KICBsYXlvdXQoDQogICAgdGl0bGUgPSAiTU88c3ViPjI8L3N1Yj4gdnMgRGlzc29sdmVkIE94eWdlbiB3aXRoIGluZGl2aWR1YWwgbGluZWFyIHJlZ3Jlc3Npb25zIiwNCiAgICB4YXhpcyA9IGxpc3QodGl0bGUgPSAiRGlzc29sdmVkIE94eWdlbiAoJSkiKSwNCiAgICB5YXhpcyA9IGxpc3QodGl0bGUgPSAiTU88c3ViPjI8L3N1Yj4gKG1nPHN1cD4tMTwvc3VwPiBPPHN1Yj4yPC9zdWI+IGc8c3VwPi0xPC9zdXA+IGg8c3VwPi0xPC9zdXA+KSIpLA0KICAgIHNob3dsZWdlbmQgPSBGQUxTRQ0KICApDQoNCiMgRGlzcGxheSBwbG90DQpwDQpgYGANCg0KKioqRmlndXJlIFMxOioqKiBpbnRlcmFjdGl2ZSBwbG90IG9mIG1ldGFib2xpYyByYXRlIG1lYXN1cmVtZW50cyAoTU/igoI7IG1nIE/igoIgZ14tMV5oXi0xXikgYnkgZGlzc29sdmVkIG94eWdlbiBwZXJjZW50YWdlIChETykgZm9yIGFsbCBmaXNoLCBpbmNsdWRpbmcgYWxsIGVzdGltYXRlcyBkdXJpbmcgdGhlIFNNUiBwaGFzZSAoaS5lLiBpbnRlcm1pdHRlbnQgcGhhc2UpLiBJbmRpdmlkdWFsIGxpbmVhciByZWdyZXNzaW9uIHdlcmUgZml0dGVkIGZvciB2aXN1YWwgcmVmZXJlbmNlLCBhbmQgZG8gbm90IHJlcHJlc2VudCB0aGUgYmVzdCBmaXR0aW5nIHJlZ3Jlc3Npb24uDQoNCiMjIyBGaWd1cmUgUzINCg0KTG9va2luZyBhdCB0aGUgZGlmZmVyZW5jZSByZXNwb25zZXMgaW4gdGhlIHR3byBzYWxpbml0eSBncm91cHMuDQoNCmBgYHtyfQ0Kc2FsaW5pdHlfc3VtbWFyeSA8LSBzbG9wZV90aWR5ICU+JSANCiAgZHBseXI6Omdyb3VwX2J5KHNhbGluaXR5X2dyb3VwKSAlPiUgDQogIGRwbHlyOjpyZWZyYW1lKG4gPSBsZW5ndGgodW5pcXVlKGlkKSkpDQoNCnNsb3BlX3RpZHkgJT4lIA0KICBnZ3Bsb3QoYWVzKHkgPSBNTzJfZywgeCA9IERPLCBjb2xvdXIgPSBpZCkpICsgIyBEZWZhdWx0IGFlc3RoZXRpY3MNCiAgZ2VvbV9wb2ludChzaG93LmxlZ2VuZCA9IEZBTFNFKSArDQogIGdlb21fc21vb3RoKGFlcyhncm91cCA9IGlkKSwgbWV0aG9kID0gImxtIiwgc2UgPSBGQUxTRSwgY29sb3VyID0gc2NhbGVzOjphbHBoYSgiYmxhY2siLCAwLjUpKSArICMgVHJhbnNwYXJlbnQgYmxhY2sgbGluZXMNCiAgZ2VvbV9zbW9vdGgobWV0aG9kID0gImxtIiwgc2UgPSBUUlVFLCBjb2xvdXIgPSAicmVkIikgKyAjIE92ZXJhbGwgc21vb3RoIGxpbmUNCiAgZ2VvbV9zbW9vdGgobWV0aG9kID0gImdhbSIsIGZvcm11bGEgPSB5IH4gcyh4LCBicyA9ICJjcyIpLCBzZSA9IFRSVUUsIGNvbG91ciA9ICJyZWQiLCBsaW5ldHlwZSA9ICJkYXNoZWQiKSArDQogIHRoZW1lX2NsZWFuKCkgKw0KICBmYWNldF93cmFwKH5zYWxpbml0eV9ncm91cCkgKw0KICBsYWJzKA0KICAgIHN1YnRpdGxlID0gIm1vMiB2cyBvMiBieSBzYWxpbml0eSB0cmVhdG1lbnQiLA0KICAgIHggPSAiRGlzc29sdmVkIG94eWdlbiBwZXJjZW50YWdlIChETykiLA0KICAgIHkgPSAiTU8yIChPMiBtZy9nL2gpIg0KICApICsNCiAgZ2VvbV90ZXh0KGRhdGEgPSBzYWxpbml0eV9zdW1tYXJ5LA0KICAgICAgICAgICAgYWVzKHggPSAtSW5mLCB5ID0gSW5mLCBsYWJlbCA9IHBhc3RlMCgiaXRhbGljKG4pID09ICIsIG4pKSwNCiAgICAgICAgICAgIGhqdXN0ID0gLTAuMSwgdmp1c3QgPSAxLjIsIGluaGVyaXQuYWVzID0gRkFMU0UsIHBhcnNlID0gVFJVRSkNCmBgYA0KDQoqKipGaWd1cmUgUzI6KioqIE1ldGFib2xpYyByYXRlIG1lYXN1cmVtZW50cyAoTU/igoI7IG1nIE/igoIgZ14tMV5oXi0xXikgYnkgZGlzc29sdmVkIG94eWdlbiBwZXJjZW50YWdlIChETykgZm9yIGZpc2ggZnJvbSB0aGUgdHdvIHNhbGluaXR5IHRyZWF0bWVudHMsIGluY2x1ZGluZyBhbGwgZXN0aW1hdGVzIGR1cmluZyB0aGUgU01SIHBoYXNlIChpLmUuIGludGVybWl0dGVudCBwaGFzZSkuIEluZGl2aWR1YWwgbGluZWFyIHJlZ3Jlc3Npb24gd2VyZSBmaXR0ZWQgZm9yIHZpc3VhbCByZWZlcmVuY2UsIGFuZCBkbyBub3QgcmVwcmVzZW50IHRoZSBiZXN0IGZpdHRpbmcgcmVncmVzc2lvbi4gVGhlIHNvbGlkIHJlZCBhbmQgZGFzaGVkIHJlZCBsaW5lIHJlcHJlc2VudHMgdGhlIGdsb2JhbCB0cmVuZCBpbiBlYWNoIG9mIHRoZSB0cmVhdG1lbnRzLCBib3RoIGEgbGluZWFyIChzb2xpZCByZWQpIGFuZCBub24tbGluZWFyIChkYXNoZWQgcmVkOyBHZW5lcmFsaXplZCBBZGRpdGl2ZSBNb2RlbCBmaXR0ZWQgd2l0aCBgZ2VvbV9zbW9vdGhgKS4NCg0KIyMjIEZpZ3VyZSBTMw0KDQpBIHBsb3QgdG8gbG9vayBhdCB0aGUgZGlmZmVyZW50IGNoYW1iZXIgdHlwZXMNCg0KYGBge3J9DQpjaGFtYmVyX3N1bW1hcnkgPC0gc2xvcGVfdGlkeSAlPiUgDQogIGRwbHlyOjpncm91cF9ieShjaGFtYmVyX3R5cGUpICU+JSANCiAgZHBseXI6OnJlZnJhbWUobiA9IGxlbmd0aCh1bmlxdWUoaWQpKSkNCg0Kc2xvcGVfdGlkeSAlPiUgDQogIGdncGxvdChhZXMoeSA9IE1PMl9nLCB4ID0gRE8sIGNvbG91ciA9IGlkKSkgKyAjIERlZmF1bHQgYWVzdGhldGljcw0KICBnZW9tX3BvaW50KHNob3cubGVnZW5kID0gRkFMU0UpICsNCiAgZ2VvbV9zbW9vdGgoYWVzKGdyb3VwID0gaWQpLCBtZXRob2QgPSAibG0iLCBzZSA9IEZBTFNFLCBjb2xvdXIgPSBzY2FsZXM6OmFscGhhKCJibGFjayIsIDAuNSkpICsgIyBUcmFuc3BhcmVudCBibGFjayBsaW5lcw0KICBnZW9tX3Ntb290aChtZXRob2QgPSAibG0iLCBzZSA9IFRSVUUsIGNvbG91ciA9ICJyZWQiKSArICMgT3ZlcmFsbCBzbW9vdGggbGluZQ0KICBnZW9tX3Ntb290aChzZSA9IFRSVUUsIGNvbG91ciA9ICJyZWQiLCBsaW5ldHlwZSA9ICJkYXNoZWQiKSArDQogIHRoZW1lX2NsZWFuKCkgKw0KICBmYWNldF93cmFwKH5jaGFtYmVyX3R5cGUsIHNjYWxlID0gImZyZWUiKSArDQogIGxhYnMoDQogICAgc3VidGl0bGUgPSAibW8yIHZzIG8yIGJ5IGNoYW1iZXIgdHlwZSIsDQogICAgeCA9ICJEaXNzb2x2ZWQgb3h5Z2VuIHBlcmNlbnRhZ2UgKERPKSIsDQogICAgeSA9ICJNTzIgKG1nIE8yIGcvaCkiDQogICkgKw0KICBnZW9tX3RleHQoZGF0YSA9IGNoYW1iZXJfc3VtbWFyeSwNCiAgICAgICAgICAgIGFlcyh4ID0gLUluZiwgeSA9IEluZiwgbGFiZWwgPSBwYXN0ZTAoIml0YWxpYyhuKSA9PSAiLCBuKSksDQogICAgICAgICAgICBoanVzdCA9IC0wLjEsIHZqdXN0ID0gMS4yLCBpbmhlcml0LmFlcyA9IEZBTFNFLCBwYXJzZSA9IFRSVUUpDQpgYGANCg0KKioqRmlndXJlIFMzOioqKiBNZXRhYm9saWMgcmF0ZSBtZWFzdXJlbWVudHMgKE1P4oKCOyBtZyBP4oKCIGdeLTFeaF4tMV4pIGJ5IGRpc3NvbHZlZCBveHlnZW4gcGVyY2VudGFnZSAoRE8pIGZvciBmaXNoIHRlc3RlZCBpbiB0aGUgNCBkaWZmZXJlbnQgY2hhbWJlciB0eXBlcywgaW5jbHVkaW5nIGFsbCBlc3RpbWF0ZXMgZHVyaW5nIHRoZSBTTVIgcGhhc2UgKGkuZS4gaW50ZXJtaXR0ZW50IHBoYXNlKS4gSW5kaXZpZHVhbCBsaW5lYXIgcmVncmVzc2lvbiB3ZXJlIGZpdHRlZCBmb3IgdmlzdWFsIHJlZmVyZW5jZSwgYW5kIGRvIG5vdCByZXByZXNlbnQgdGhlIGJlc3QgZml0dGluZyByZWdyZXNzaW9uLiBUaGUgc29saWQgcmVkIGFuZCBkYXNoZWQgcmVkIGxpbmUgcmVwcmVzZW50cyB0aGUgZ2xvYmFsIHRyZW5kIGluIGVhY2ggb2YgdGhlIHRyZWF0bWVudHMsIGJvdGggYSBsaW5lYXIgKHNvbGlkIHJlZCkgYW5kIG5vbi1saW5lYXIgKGRhc2hlZCByZWQ7IEdlbmVyYWxpemVkIEFkZGl0aXZlIE1vZGVsIGZpdHRlZCB3aXRoIGBnZW9tX3Ntb290aGApLg0KDQoNCiMjIFJvdXRpbmUgTU/igoINCg0KTWFraW5nIGFuIFNNUiBwaGFzZSBvbmx5IGRhdGEgZnJhbWUNCg0KYGBge3J9DQpzbG9wZV90aWR5X3NtciA8LSBzbG9wZV90aWR5ICU+JSANCiAgZHBseXI6OmZpbHRlcihwaGFzZSA9PSAic21yIikNCmBgYA0KDQojIyMgRmlndXJlIFM0DQoNClJvdXRpbmUgTU88c3ViPjI8L3N1Yj4gYnkgc2FsaW5pdHkNCg0KYGBge3J9DQptZWFuX21vMl9zYWxpbml0eSA8LSBzbG9wZV90aWR5X3NtciAlPiUgDQogIGRwbHlyOjpncm91cF9ieShzYWxpbml0eV9ncm91cCkgJT4lIA0KICBkcGx5cjo6cmVmcmFtZShtZWFuX21vMiA9IG1lYW4oTU8yLCBuYS5ybSA9IFRSVUUpKQ0KDQpmaWdfaSA8LSBnZ3Bsb3QoKSArDQogICAgZ2VvbV92aW9saW4oZGF0YSA9IHNsb3BlX3RpZHlfc21yLCBhZXMoeCA9IHNhbGluaXR5X2dyb3VwLCB5ID0gTU8yLCBmaWxsID0gc2FsaW5pdHlfZ3JvdXApLCBjb2xvciA9IE5BLCBhbHBoYSA9IDAuMykgKw0KICBnZW9tX2ppdHRlcihkYXRhID0gc2xvcGVfdGlkeV9zbXIsIGFlcyh4ID0gc2FsaW5pdHlfZ3JvdXAsIHkgPSBNTzIsIGZpbGwgPSBzYWxpbml0eV9ncm91cCksDQogICAgICAgICAgICAgICAgICAgICAgIHNoYXBlID0gMjEsIHNpemUgPSAyLCBjb2xvciA9ICJibGFjayIsIGFscGhhID0gMC4yKSArDQogIGdlb21fYm94cGxvdChkYXRhID0gc2xvcGVfdGlkeV9zbXIsIGFlcyh4ID0gc2FsaW5pdHlfZ3JvdXAsIHkgPSBNTzIsIGZpbGwgPSBzYWxpbml0eV9ncm91cCksDQogICAgICAgICAgICAgICAgICAgICAgICBzaXplID0gMSwgYWxwaGEgPSAwLjUsIG91dGxpZXIuc2hhcGUgPSBOQSwgd2lkdGggPSAwLjMpICsNCiAgZ2VvbV9wb2ludChkYXRhID0gbWVhbl9tbzJfc2FsaW5pdHksIA0KICAgICAgICAgICAgICAgIGFlcyh4ID0gc2FsaW5pdHlfZ3JvdXAsIHkgPSBtZWFuX21vMiwgZmlsbCA9IHNhbGluaXR5X2dyb3VwKSwgDQogICAgICAgICAgICAgICAgc2l6ZSA9IDMsIGFscGhhID0gMC44LCBjb2xvdXIgPSAiYmxhY2siLCBzdHJva2UgPSAyKSArDQogIHNjYWxlX2ZpbGxfbWFudWFsKHZhbHVlcyA9IGMoIiM0QjUzMjAiLCAiIzAwMDA4MCIpKSArICAjIEN1c3RvbSBmaWxsIGNvbG91cnMNCiAgc2NhbGVfY29sb3VyX21hbnVhbCh2YWx1ZXMgPSBjKCIjNEI1MzIwIiwgIiMwMDAwODAiKSkgKw0KICB0aGVtZV9jbGVhbigpICsNCiAgdGhlbWUobGVnZW5kLnBvc2l0aW9uID0gIm5vbmUiKSArDQogIGxhYnMoDQogICAgc3VidGl0bGUgPSAiIiwNCiAgICB4ID0gIlNhbGluaXR5IGdyb3VwIChwcHQpIiwNCiAgICB5ID0gIlJvdXRpbmUgTU8yIChtZyBPMiBnL2gpIg0KICApDQoNCmZpZ19pDQpgYGANCg0KKipGaWd1cmUgUzQ6KiogUGxvdCBvZiBhbGwgTU88c3ViPjI8L3N1Yj4gbWVhc3VyZXMgZHVyaW5nIFNNUiBwaGFzZSBieSBzYWxpbml0eSB0cmVhdG1lbnQuIFRoZSBzbWFsbCBwb2ludHMgYXJlIHRoZSByYXcgb2JzZXJ2ZWQgdmFsdWVzLCB0aGUgc2hhZGVkIGFyZWEgYmVoaW5kIHRoZSBwb2ludHMgaXMgdGhlIGEga2VybmVsIGRlbnNpdHkgb2YgdGhlIG9ic2VydmVkIGRhdGEsIHRoZSBib3ggcGxvdCBzaG93cyB0aGUgbWVkaWFuIGFuZCBpbnRlcnF1YXJ0aWxlIHJhbmdlIChJUVIpLCBhbmQgdGhlIGxhcmdlIHBvaW50IHNob3dzIHRoZSBtZWFuLg0KDQojIyBTTVINCg0KIyMjIEZpZ3VyZSBTNQ0KDQpIZXJlJ3MgdGhlIHNhbWUgcGxvdCBidXQgZm9yIG9ubHkgdGhlIFNNUiwgYXMgZXN0aW1hdGVkIHdpdGggY2FsY1NNUiBmdW5jdGlvbiBieSBDaGFib3QsIFN0ZWZmZW5zZW4gYW5kIEZhcnJlbGwgKDIwMTYpXlsxXV4uIFNwZWNpZmljYWxseSwgd2UgdXNlIG1lYW4gb2YgdGhlIGxvd2VzdCBub3JtYWwgZGlzdHJpYnV0aW9uIChNTE5EKSB3aGVyZSBDVm1sbmQgXDwgNS40IGFuZCB0aGUgbWVhbiBvZiB0aGUgbG93ZXIgMjAlIHF1YW50aWxlIChxMC4yKSB3ZXJlIENWbWxuZCBcPiA1LjQuIElmIENWbWxuZCBpcyBub3QgY2FsY3VsYXRlZCB3ZSBoYXZlIHVzZWQgcTAuMi4NCg0KYGBge3J9DQpzbXJfb25seSA8LSBzbG9wZV90aWR5X3NtciAlPiUgDQogIGRwbHlyOjpncm91cF9ieShpZCkgJT4lIA0KICBkcGx5cjo6c2xpY2UoMSkNCg0KbWVhbl9zbXJfc2FsaW5pdHkgPC0gc21yX29ubHkgJT4lIA0KICBkcGx5cjo6Z3JvdXBfYnkoc2FsaW5pdHlfZ3JvdXApICU+JSANCiAgZHBseXI6OnJlZnJhbWUobWVhbl9zbXIgPSBtZWFuKFNNUiwgbmEucm0gPSBUUlVFKSkNCg0KZmlnX2kgPC0gZ2dwbG90KCkgKw0KICAgIGdlb21fdmlvbGluKGRhdGEgPSBzbXJfb25seSwgYWVzKHggPSBzYWxpbml0eV9ncm91cCwgeSA9IFNNUiwgZmlsbCA9IHNhbGluaXR5X2dyb3VwKSwgY29sb3IgPSBOQSwgYWxwaGEgPSAwLjMpICsNCiAgZ2VvbV9qaXR0ZXIoZGF0YSA9IHNtcl9vbmx5LCBhZXMoeCA9IHNhbGluaXR5X2dyb3VwLCB5ID0gU01SLCBmaWxsID0gc2FsaW5pdHlfZ3JvdXApLA0KICAgICAgICAgICAgICAgICAgICAgICBzaGFwZSA9IDIxLCBzaXplID0gMiwgY29sb3IgPSAiYmxhY2siLCBhbHBoYSA9IDAuMikgKw0KICBnZW9tX2JveHBsb3QoZGF0YSA9IHNtcl9vbmx5LCBhZXMoeCA9IHNhbGluaXR5X2dyb3VwLCB5ID0gU01SLCBmaWxsID0gc2FsaW5pdHlfZ3JvdXApLA0KICAgICAgICAgICAgICAgICAgICAgICAgc2l6ZSA9IDEsIGFscGhhID0gMC41LCBvdXRsaWVyLnNoYXBlID0gTkEsIHdpZHRoID0gMC4zKSArDQogIGdlb21fcG9pbnQoZGF0YSA9IG1lYW5fc21yX3NhbGluaXR5LCANCiAgICAgICAgICAgICAgICBhZXMoeCA9IHNhbGluaXR5X2dyb3VwLCB5ID0gbWVhbl9zbXIsIGZpbGwgPSBzYWxpbml0eV9ncm91cCksIA0KICAgICAgICAgICAgICAgIHNpemUgPSAzLCBhbHBoYSA9IDAuOCwgY29sb3VyID0gImJsYWNrIiwgc3Ryb2tlID0gMikgKw0KICBzY2FsZV9maWxsX21hbnVhbCh2YWx1ZXMgPSBjKCIjNEI1MzIwIiwgIiMwMDAwODAiKSkgKyAgIyBDdXN0b20gZmlsbCBjb2xvdXJzDQogIHNjYWxlX2NvbG91cl9tYW51YWwodmFsdWVzID0gYygiIzRCNTMyMCIsICIjMDAwMDgwIikpICsNCiAgdGhlbWVfY2xlYW4oKSArDQogIHRoZW1lKGxlZ2VuZC5wb3NpdGlvbiA9ICJub25lIikgKw0KICBsYWJzKA0KICAgIHN1YnRpdGxlID0gIiIsDQogICAgeCA9ICJTYWxpbml0eSBncm91cCAocHB0KSIsDQogICAgeSA9ICJNTzIgKG1nIE8yIGcvaCkiDQogICkNCg0KZmlnX2kNCmBgYA0KDQoqKkZpZ3VyZSBTNToqKiBUaGUgc3RhbmRhcmQgbWV0YWJvbGljIHJhdGUgKFNNUikgYnkgc2FsaW5pdHkgdHJlYXRtZW50LiBUaGUgc21hbGwgcG9pbnRzIGFyZSB0aGUgcmF3IG9ic2VydmVkIHZhbHVlcywgdGhlIHNoYWRlZCBhcmVhIGJlaGluZCB0aGUgcG9pbnRzIGlzIHRoZSBhIGtlcm5lbCBkZW5zaXR5IG9mIHRoZSBvYnNlcnZlZCBkYXRhLCB0aGUgYm94IHBsb3Qgc2hvd3MgdGhlIG1lZGlhbiBhbmQgaW50ZXJxdWFydGlsZSByYW5nZSAoSVFSKSwgYW5kIHRoZSBsYXJnZSBwb2ludCBzaG93cyB0aGUgbWVhbi4NCg0KIyMgSW5kaXZpZHVhbCBP4oKCLCBNT+KCgiwgYW5kIFNNUg0KDQpIZXJlIHdlIHdpbGwgcGxvdCB0aGUgaW5kaXZpZHVhbCByZWxhdGlvbnNoaXAgYmV0d2VlbiBP4oKCLCBNT+KCgiwgYW5kIFNNUiBmb3IgYWxsIGZpc2gNCg0KQ3JlYXRlIG91dHB1dCBkaXJlY3RvcnkgaWYgbmVlZGVkDQoNCmBgYHtyLCBlY2hvPUZBTFNFfQ0KIyBDcmVhdGUgb3V0cHV0IGRpcmVjdG9yeSBpZiBuZWVkZWQNCm91dHB1dF9maWdfc2xvcGVzX3dkIDwtIGZpbGUucGF0aChvdXRwdXRfZmlnX3dkLCAic2xvcGVzIikNCmlmICghZGlyLmV4aXN0cyhvdXRwdXRfZmlnX3Nsb3Blc193ZCkpIHsNCiAgZGlyLmNyZWF0ZShvdXRwdXRfZmlnX3Nsb3Blc193ZCkNCn0NCmBgYA0KDQpMb29wIHRocm91Z2ggZWFjaCBmaXNoIElEIHRvIGNyZWF0ZSBhIHBsb3QsIHNhdmUgdGhlc2UgdG8gYSBzaW5nbGUgUERGIGZpbGUsIHRoaXMgaXMgY2FsbGVkIGBjb21iaW5lZF9zbG9wZXMucGRmYCAgIA0KDQpgYGB7ciwgZmlnLmhlaWdodCA9IDMsIGZpZy53aWR0aD0gNH0NCiMgIyBEZWZpbmUgY29uc2lzdGVudCBjb2xvcnMgZm9yIGFsbCBwaGFzZXMNCiMgcGhhc2VfY29sb3JzIDwtIGMoDQojICAgIjUwYyIgID0gIiNlNTZiNmYiLA0KIyAgICI3NWMiICA9ICIjYjU2NTc2IiwNCiMgICAiMTAwYyIgPSAiIzZkNTk3YSIsDQojICAgInNtciIgID0gIiMzNTUwNzAiDQojICkNCiMgDQojIGlkcyA8LSBzbG9wZV90aWR5ICU+JSANCiMgICBkcGx5cjo6ZGlzdGluY3QoaWQpICU+JSANCiMgICBwdWxsKGlkKSAlPiUgDQojICAgYXMubGlzdCgpDQojIA0KIyBNTzJfcGxvdF9saXN0IDwtIGxpc3QoKQ0KIyANCiMgIyAxKSBPcGVuIHRoZSBQREYgZGV2aWNlIG9uY2UNCiMgcGRmKA0KIyAgIGZpbGUgICA9IGZpbGUucGF0aChvdXRwdXRfZmlnX3Nsb3Blc193ZCwgImNvbWJpbmVkX3Nsb3Blcy5wZGYiKSwgDQojICAgd2lkdGggID0gOCwgDQojICAgaGVpZ2h0ID0gNg0KIyApDQojIA0KIyAjIDIpIExvb3Agb3ZlciBJRHMgYW5kIGNyZWF0ZSBlYWNoIHBsb3QNCiMgZm9yIChpZF9pIGluIGlkcykgew0KIyAgIA0KIyAgIHNtciA8LSBzbG9wZV90aWR5ICU+JSANCiMgICAgIGRwbHlyOjpmaWx0ZXIoaWQgPT0gaWRfaSkgJT4lIA0KIyAgICAgZHBseXI6OnNsaWNlKDEpICU+JSANCiMgICAgIGRwbHlyOjpwdWxsKFNNUikNCiMgICANCiMgICBwbG90IDwtIHNsb3BlX3RpZHkgJT4lIA0KIyAgICAgZHBseXI6OmZpbHRlcihpZCA9PSBpZF9pKSAlPiUgDQojICAgICBnZ3Bsb3QoYWVzKHggPSBvMiwgeSA9IE1PMikpICsNCiMgICAgIGdlb21faGxpbmUoeWludGVyY2VwdCA9IHNtciwgbGluZXR5cGUgPSAiZGFzaGVkIiwgY29sb3IgPSAiZGFya3JlZCIpICsNCiMgICAgIGdlb21fcG9pbnQoYWVzKGNvbG91ciA9IHBoYXNlKSkgKw0KIyAgICAgc2NhbGVfY29sb3JfbWFudWFsKHZhbHVlcyA9IHBoYXNlX2NvbG9ycywgZHJvcCA9IEZBTFNFKSArICAjIEVuc3VyZXMgY29uc2lzdGVudCBjb2xvdXJzDQojICAgICB0aGVtZV9jbGVhbigpICsNCiMgICAgIGxhYnMoDQojICAgICAgIHN1YnRpdGxlID0gcGFzdGUwKGlkX2ksICIgc2xvcGVzIiksDQojICAgICAgIHggPSAiTWVhbiBvMiAobWdfcGVyX2wpIiwNCiMgICAgICAgeSA9ICJhYnMobW8yKSAobWdfcGVyX2wpIg0KIyAgICAgKQ0KIyAgIA0KIyAgICMgSW5zdGVhZCBvZiBzYXZpbmcgZWFjaCBwbG90IHNlcGFyYXRlbHksIGp1c3QgcHJpbnQgaXQNCiMgICBwcmludChwbG90KQ0KIyAgIA0KIyAgIE1PMl9wbG90X2xpc3RbW2lkX2ldXSA8LSBwbG90DQojIH0NCiMgDQojICMgMykgQ2xvc2UgdGhlIFBERiBkZXZpY2UgKmFmdGVyKiB0aGUgbG9vcA0KIyBkZXYub2ZmKCkNCmBgYA0KDQoqKkZpZ3VyZSBTNyoqOiBNZXRhYm9saWMgcmF0ZSBtZWFzdXJlbWVudHMgKE1P4oKCOyBtZyBP4oKCIGdeLTFeaF4tMV4gYnkgTzxzdWI+Mjwvc3ViPikgYXQgZWFjaCBvZiB0aGUgZm91ciBleHBlcmltZW50YWwgcGhhc2UgKHRoZSBvdmVyIG5pZ2h0IFNNUiBpbnRlcm1pdHRlbnQtZmxvdyByZXNwaXJvbWV0cnkgcGhhc2UsIGNsb3NlZCBwaGFzZXMgYXQgNTAlLCA3NSUgb3IgMTAwJSBPPHN1Yj4yPC9zdWI+KS4gVGhlIGVzdGltYXRlZCBTTVIgdmFsdWUgaXMgcmVwcmVzZW50ZWQgYXMgYSBkYXNoZWQgcmVkIGxpbmUuDQoNCjwhLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLT4NCg0KIyDwn6euIEFuYWx5c2lzDQoNCjwhLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLT4NCg0KIyMgUm91dGluZSBNTzINCg0KIyMjIFNjYWxpbmcgcHJlZGljdG9ycw0KDQpIZXJlIHdlIHNjYWxlIG91ciBwcmVkaWN0b3JzIGZvciB0aGUgbW9kZWwuIFdlIGhhdmUgZmlyc3QgYXBwbGllZCBhbiBsb2cgdHJhbnNmb3JtYXRpb24gdG8gbWFzcywgc28gaXRzIGRpcmVjdGx5IGludGVycHJldGFibGUgYXMgdGhlIG1hc3Mtc2NhbGluZyBleHBvbmVudCwganVzdCBsaWtlIGluIG1ldGFib2xpYyB0aGVvcnkuDQoNCmBgYHtyfQ0Kc2NhbGVfbGlzdCA8LSBjKCJ0ZW1wIiwgIm9yZGVyIiwgImxvZ19tYXNzIikgDQoNCnNsb3BlX3RpZHlfc21yIDwtIHNsb3BlX3RpZHlfc21yICU+JQ0KICBkcGx5cjo6bXV0YXRlKGxvZ19tYXNzID0gbG9nKG1hc3MpLA0KICAgICAgICAgICAgICAgIGFjcm9zcyhhbGxfb2Yoc2NhbGVfbGlzdCksIH4gc2NhbGUoLngsIGNlbnRlciA9IFRSVUUsIHNjYWxlID0gRkFMU0UpLA0KICAgICAgICAgICAgICAgICAgICAgICAubmFtZXMgPSAiey5jb2x9X2MiKSwNCiAgICAgICAgICAgICAgICBsaWdodF9kYXJrX2MgPSBpZl9lbHNlKGxpZ2h0X2RhcmsgPT0gImxpZ2h0IiwgMC41LCAtMC41KSkNCmBgYA0KDQpTdW1tYXJ5IG9mIHByZWRpY3RvcnMgdXNlZCBpbiB0aGUgbW9kZWwgYmVsb3cNCg0KYGBge3J9DQpzbG9wZV90aWR5X3NtciAlPiUNCiAgZHBseXI6Om11dGF0ZShjeWNsZSA9IGFzLm51bWVyaWMoY3ljbGUpKSAlPiUgDQogIGRwbHlyOjpyZWZyYW1lKHRlbXBfcmFuZ2UgPSBwYXN0ZTAocm91bmQobWluKHRlbXAsIG5hLnJtID0gVFJVRSksMiksICLigJMiLCByb3VuZChtYXgodGVtcCwgbmEucm0gPSBUUlVFKSwyKSksDQogICAgICAgICAgICBtYXNzX3JhbmdlID0gcGFzdGUwKHJvdW5kKG1pbihtYXNzLCBuYS5ybSA9IFRSVUUpLDIpLCAi4oCTIiwgcm91bmQobWF4KG1hc3MsIG5hLnJtID0gVFJVRSksMikpLA0KICAgICAgICAgIGN5Y2xlX3JhbmdlID0gcGFzdGUwKHJvdW5kKG1pbihjeWNsZSwgbmEucm0gPSBUUlVFKSwyKSwgIuKAkyIsIHJvdW5kKG1heChjeWNsZSwgbmEucm0gPSBUUlVFKSwyKSksDQogICAgICAgICAgbnVtYmVyX2Zpc2ggPSBsZW5ndGgodW5pcXVlKGlkKSkgIA0KICAgICAgICAgICkgJT4lIA0KICBndCgpDQpgYGANCg0KIyMjIE1vZGVsIHN0cnVjdHVyZQ0KDQpIZXJlIHdlIHdpbGwgdXNlIGEgQmF5ZXNpYW4gR2VuZXJhbGlzZWQgTGluZWFyIE1peGVkIE1vZGVsIChHTE1NKSB3aXRoIGEgR2FtbWEgZGlzdHJpYnV0aW9uIGFuZCBhIGxvZyBsaW5rLCB3aGVyZSB0aGUgc2hhcGUgcGFyYW1ldGVyICjOsSkgaXMgYWxzbyBtb2RlbGxlZCBhcyBhIGZ1bmN0aW9uIG9mIHByZWRpY3RvcnMuIFRoaXMgbW9kZWxzIE1PPHN1Yj4yPC9zdWI+IGJ5IHNhbGluaXR5IGR1cmluZyB0aGUgU01SIHBoYXNlIHRvIHNlZSBpZiB0aGUgZmlzaCBoZWxkIGF0IGRpZmZlcmVudCBzYWxpbml0aWVzIGhhdmUgZGlmZmVyZW50IFJNUnMuIFdlIGhhdmUgYWxzbyBhZGRlZCBhIGZldyBzY2FsZWQgcHJlZGljdG9ycywgdGhhdCBtYXkgaGVscCBkZXNjcmliZSB2YXJpYXRpb24gaW4gdGhlIGRhdGEsIHN1Y2ggYXMgbWFzcyAoZzsgMC4yMS0tMS42KSB0ZW1wZXJhdHVyZSAowrBDOyAxMy44NC0tMTQuMzgpLCBjeWNsZSBvcmRlciAoNS0tMjcpLCBhbmQgbGlnaHQvZGFyayBjeWNsZSAobGlnaHQgb3IgZGFyazsgbGlnaHQgYmV0d2VlbiAwNzowMDowMCBhbmQgMTk6MDA6MDApLCB3ZSBhbHNvIGluY2x1ZGUgYSByYW5kb20gZWZmZWN0IGZvciBmaXNoIGlkIHRvIGFjY291bnQgZm9yIG11bHRpcGxlIE1PPHN1Yj4yPC9zdWI+IG1lYXN1cmVzIG9uIGVhY2ggZmlzaCAoMS0tNTgpLiBXZSBhbGxvd2VkIHRoZSB0aGUgc2hhcGUgcGFyYW1ldGVyICjOsSkgdG8gdmFyeSBhcyBhIGZ1bmN0aW9uIG9mIHNvbWUgb2YgdGhlIHByZWRpY3RvcnMgKGUuZy4gc2FsaW5pdHlfZ3JvdXAsIG9yZGVyX3opIHRvIGltcHJvdmUgZml0Lg0KDQpgYGB7cn0NCm1vMl9nYW1tYV9iZiA8LSBiZihNTzIgfiB0ZW1wX2MgKyANCiAgICAgICAgICAgICAgICAgICAgIG9yZGVyX2MgICsgDQogICAgICAgICAgICAgICAgICAgICBsaWdodF9kYXJrX2MgKyANCiAgICAgICAgICAgICAgICAgICAgIGxvZ19tYXNzX2MgKyANCiAgICAgICAgICAgICAgICAgICAgIHNhbGluaXR5X2dyb3VwICsgKDF8aWQpLA0KICAgICAgICAgICAgICAgICAgIHNoYXBlIH4gc2FsaW5pdHlfZ3JvdXAgKyANCiAgICAgICAgICAgICAgICAgICAgIG9yZGVyX2MsDQogICAgICAgICAgICAgICAgICAgZmFtaWx5ID0gR2FtbWEobGluayA9ICJsb2ciKSkNCmBgYA0KDQojIyMgUHJpb3Igc2VsZWN0aW9uDQoNClRoZXNlIGFyZSB0aGUgZGVmYXVsdCBwcmlvcnMuIFdlIHdpbGwgdXNlIHRoZXNlLg0KDQpgYGB7cn0NCmRlZmF1bHRfcHJpb3IgPC0gZ2V0X3ByaW9yKG1vMl9nYW1tYV9iZiwgZGF0YSA9IHNsb3BlX3RpZHlfc21yLCBmYW1pbHkgPSBHYW1tYShsaW5rID0gImxvZyIpKQ0KZGVmYXVsdF9wcmlvciAlPiUgDQogIGd0KCkNCmBgYA0KDQojIyMgUnVuIG1vZGVsDQoNCkhlcmUgd2UgcnVuIHRoZSBtb2RlbCwgSSBoYXZlIGhhc2hlZCB0aGlzIG91dCBiZWNhdXNlIEkgaGF2ZSBzYXZlZCB0aGUgbW9kZWwgZm9yIHF1aWNrIHJlbG9hZGluZy4NCg0K4o+t77iPICoqU2tpcCB0aGlzIHN0ZXAqKiBpZiB5b3UgaGF2ZSBkb3dubG9hZGVkIHRoZSBzYXZlZCBtb2RlbCAnbW8yX21vZF9nYW1tYS5yZHMnDQoNCmBgYHtyLCBlY2hvPUZBTFNFfQ0KIyBtbzJfbW9kX2dhbW1hIDwtIGJybShtbzJfZ2FtbWFfYmYsDQojICAgICAgICAgICAgICAgIGRhdGEgPSBzbG9wZV90aWR5X3NtciwNCiMgICAgICAgICAgICAgICAgY29yZXMgPSA0LA0KIyAgICAgICAgICAgICAgICBjaGFpbnMgPSA0LA0KIyAgICAgICAgICAgICAgICBwcmlvciA9IGRlZmF1bHRfcHJpb3IsDQojICAgICAgICAgICAgICAgIHdhcm11cCA9IDEwMDAsDQojICAgICAgICAgICAgICAgIHNlZWQgPSAxNDMwMTksDQojICAgICAgICAgICAgICAgIHRoaW4gPSAyLA0KIyAgICAgICAgICAgICAgICBpdGVyID0gODAwMCwNCiMgICAgICAgICAgICAgICAgc2F2ZV9wYXJzID0gc2F2ZV9wYXJzKGFsbD1UUlVFKSwNCiMgICAgICAgICAgICAgICAgc2FtcGxlX3ByaW9yID0gVFJVRSwNCiMgICAgICAgICAgICAgICAgZmlsZSA9IHBhc3RlMChvdXRwdXRfbW9kc193ZCwgJy4vbW8yX21vZF9nYW1tYScpKQ0KIyBwcmludCgiTW9kZWwgY29tcGxldGUiKQ0KYGBgDQoNCkhlcmUgd2UgcmVsb2FkIHRoZSBtb2RlbA0KDQpgYGB7cn0NCm1vMl9tb2RfZ2FtbWEgPC0gIHJlYWRSRFMoZmlsZSA9IHBhc3RlMChvdXRwdXRfbW9kc193ZCwgJy4vbW8yX21vZF9nYW1tYS5yZHMnKSkNCmBgYA0KDQojIyMgTW9kZWwgZGlhZ25vc3RpY3MNCg0KQ2hlY2tpbmcgbW9kZWwgY29udmVyZ2VuY2UNCg0KYGBge3J9DQpjb2xvcl9zY2hlbWVfc2V0KCJyZWQiKQ0KcGxvdChtbzJfbW9kX2dhbW1hLCBhc2sgPSBGKQ0KYGBgDQoNCkNoZWNraW5nIHJoYXQgYXJlIGVxdWFsIHRvIG9uZQ0KDQpgYGB7cn0NCnN1bW1hcnkobW8yX21vZF9nYW1tYSkNCmBgYA0KDQpVc2luZyBsZWF2ZSBvbmUgb3V0IChsb28pIG1lYXN1cmUgb2YgZml0LCB0aGUgbW9kZWwgYXBwZWFycyB0byBwcmVmb3JtIHdlbGwsIGFsbCwgYnV0IG9uZSBQYXJldG8gayBlc3RpbWF0ZXMgYXJlIGdvb2QgKGsgXDwgMC43KQ0KDQpgYGB7cn0NCmxvbyhtbzJfbW9kX2dhbW1hKQ0KYGBgDQoNCk1vZGVsIHByZWRpY3Rpb25zIGdlbmVyYWxseSBhbGlnbiB3aXRoIHRoZSBvYnNlcnZlZCBkYXRhDQoNCmBgYHtyfQ0KY29sb3Jfc2NoZW1lX3NldCgicmVkIikNCnAgPC0gcHBfY2hlY2sobW8yX21vZF9nYW1tYSwgdHlwZSA9ICJkZW5zX292ZXJsYXkiKQ0KcA0KYGBgDQoNCiMjIyDwn5OIIFJlc3VsdHMNCg0KV2UgZGlkIG5vdCBzZWUgYSBtZWFuaW5nZnVsIGRpZmZlcmVuY2UgYmV0d2VlbiB0aGUgcm91dGluZSBtZXRhYm9saWMgcmF0ZSBmb3IgZmlzaCBmcm9tIHRoZSB0d28gc2FsaW5pdHkgdHJlYXRtZW50cy4NCg0KIyMjIyBUYWJsZSBTMQ0KDQoqKlRhYmxlIFMxKio6IEZpeGVkIGVmZmVjdCBFc3RpbWF0ZXMgKM6yKSBhbmQgOTUlIENyZWRpYmxlIEludGVydmFscyAoOTUlIENJKQ0KDQpgYGB7cn0NCm1vZGVsX2VzdCA8LSBmaXhlZihtbzJfbW9kX2dhbW1hLCBwcm9icyA9IGMoMC4wMjUsIDAuOTc1KSkgJT4lIA0KICBhcy5kYXRhLmZyYW1lKCkgJT4lIA0KICB0aWJibGU6OnJvd25hbWVzX3RvX2NvbHVtbih2YXIgPSAiUHJlZGljdG9yIikgJT4lIA0KICBkcGx5cjo6bXV0YXRlKCfOsicgPSByb3VuZChFc3RpbWF0ZSwgMyksDQogICAgICAgICAgICAgICAgUTIuNSA9IHJvdW5kKFEyLjUsIDMpLA0KICAgICAgICAgICAgICAgIFE5Ny41ID0gcm91bmQoUTk3LjUsIDMpLA0KICAgICAgICAgICAgICAgICc5NSUgQ0knID0gcGFzdGUwKCJbIiwgUTIuNSwgIiwgIiwgUTk3LjUsICJdIikpDQoNCm1vZGVsX2VzdCAlPiUgDQogIGRwbHlyOjpzZWxlY3QoUHJlZGljdG9yLCAnzrInLCAnOTUlIENJJykgJT4lIA0KICBndCgpDQpgYGANCg0KTG9va2luZyBhdCB0aGUgbWFyZ2luYWwgbWVhbiBNTzIgZm9yIGVhY2ggc2FsaW5pdHkgdHJlYXRtZW50DQoNCmBgYHtyfQ0KZW1fcmVzdWx0cyA8LSBlbW1lYW5zKG1vMl9tb2RfZ2FtbWEsIH4gc2FsaW5pdHlfZ3JvdXApIA0KDQplbV9yZXN1bHRzX2RmIDwtIGVtX3Jlc3VsdHMgJT4lDQogIGFzLmRhdGEuZnJhbWUoKSAlPiUNCiAgbXV0YXRlKGFjcm9zcyh3aGVyZShpcy5udW1lcmljKSwgfiByb3VuZChleHAoLiksMykpKQ0KDQplbV9yZXN1bHRzX2RmICU+JSANCiAgZ3QoKQ0KYGBgDQoNClRoZXNlIGFyZSB0aGUgZXN0aW1hdGVkIG1lYW4gY29uc3RyYXN0LCB3ZSBoYXZlIGV4cG9uZW50aWF0ZWQgdGhlIGxvZyBlc3RpbWF0ZSB0byBnZXQgdGhlIGZvbGQtY2hhbmdlLiBUaGlzIG1lYW5zIHRoZSBtb2RlbCBwcmVkaWN0cyB0aGF0IHNhaWxpYW50eSBncm91cCAwIGhhcyBcfjEyJSBoaWdoZXIgbWV0YWJvbGljIHJhdGUgdGhhbiBncm91cCA5ICgxLjEyKS0tLWJ1dCB0aGlzIGRpZmZlcmVuY2UgaXMgbm90IGNyZWRpYmxlLCBiZWNhdXNlIHRoZSBpbnRlcnZhbCBmb3IgdGhhdCBkaWZmZXJlbmNlIGluY2x1ZGVzIG5vIGRpZmZlcmVuY2UgKGkuZS4gYSByYXRpbyBvZiAxLjApLg0KDQpgYGB7cn0NCmNvbnRyYXN0X3Jlc3VsdHMgPC0gY29udHJhc3QoZW1fcmVzdWx0cywgbWV0aG9kID0gInBhaXJ3aXNlIikNCg0KY29udHJhc3RfcmVzdWx0c19kZiA8LSAgY29udHJhc3RfcmVzdWx0cyAlPiUgDQogIGFzLmRhdGEuZnJhbWUoKSAlPiUNCiAgbXV0YXRlKGFjcm9zcyh3aGVyZShpcy5udW1lcmljKSwgfiByb3VuZChleHAoLiksMikpKQ0KDQpjb250cmFzdF9yZXN1bHRzX2RmICU+JSANCiAgZ3QoKQ0KYGBgDQoNCkVFTSBjb250cmFzdHMgZm9yIHRoZSB0d28gbGlnaHQgcGhhc2VzLCB3aGVyZSAtMC41IGlzIGRhcmssIGFuZCAwLjUgaXMgbGlnaHQNCg0KYGBge3J9DQplbV9yZXN1bHRzIDwtIGVtbWVhbnMobW8yX21vZF9nYW1tYSwgfiBsaWdodF9kYXJrX2MsDQogICAgICAgICAgICAgICAgICAgICAgIGF0ID0gbGlzdChsaWdodF9kYXJrX2MgPSBjKDAuNSwgLTAuNSkpKQ0KDQpjb250cmFzdF9yZXN1bHRzIDwtIGNvbnRyYXN0KGVtX3Jlc3VsdHMsIG1ldGhvZCA9ICJwYWlyd2lzZSIsIC5yZXYgPSBUUlVFKQ0KDQpjb250cmFzdF9yZXN1bHRzX2RmIDwtICBjb250cmFzdF9yZXN1bHRzICU+JSANCiAgYXMuZGF0YS5mcmFtZSgpICU+JQ0KICBtdXRhdGUoYWNyb3NzKHdoZXJlKGlzLm51bWVyaWMpLCB+IHJvdW5kKGV4cCguKSwyKSkpDQoNCmNvbnRyYXN0X3Jlc3VsdHNfZGYgJT4lIA0KICBndCgpDQpgYGANCg0KSGVyZSB3ZSB3ZSBjYWxjdWxhdGUgdGhlIG1vZGVsLWVzdGltYXRlZCBtYXJnaW5hbCBtZWFucyBmb3IgTU8yIGZvciBhIHVuaXQgaW5jcmVhc2UgaW4gbWFzcyBhbmQgdHJhbnNmb3JtIHRoaXMgb250byBhIGZvbGQtY2hhbmdlIHNjYWxlLiBXaGVyZSBsb2cobWFzcykgaGFzIGJlIGNlbnRlcmVkLCBhIGNhdWxlIG9mIDAgcmVwcmVzZW50cyB0aGUgbWVhbiBtYXNzIG9mIGZpc2gsIGFuZCAxIGNvcnJlc3BvbmRzIHRvIGEgZmlzaCB3aXRoIGxvZyhtYXNzKSA9IDEgdW5pdCBhYm92ZSB0aGUgbWVhbiAoaS5lLiAyLjcgdGltZXMgbGFyZ2UpLg0KDQpsb2cobWFzczxzdWI+Mjwvc3ViPikgPSBsb2cobWFzczxzdWI+MTwvc3ViPikrMQ0KDQpJZiB5b3UgZXhwb25lbnRpYXRlIGJvdGggc2lkZXM6DQoNCm1hc3M8c3ViPjI8L3N1Yj4gPSBtYXNzPHN1Yj4xPC9zdWI+IHggZXhwKDEpDQoNCm1hc3M8c3ViPjI8L3N1Yj4g4omIIG1hc3M8c3ViPjE8L3N1Yj4geCAyLjcxOA0KDQpgYGB7cn0NCmVtX21hc3MgPC0gZW1tZWFucyhtbzJfbW9kX2dhbW1hLCB+IGxvZ19tYXNzX2MsIGF0ID0gbGlzdChsb2dfbWFzc19jID0gYygwLCAxKSkpDQoNCiMgR2V0IHRoZSBjb250cmFzdCAoZGlmZmVyZW5jZSBpbiBsb2cgTU/igoIpLCB0aGVuIGJhY2stdHJhbnNmb3JtDQpjb250cmFzdChlbV9tYXNzLCBtZXRob2QgPSAicmV2cGFpcndpc2UiKSAlPiUNCiAgYXMuZGF0YS5mcmFtZSgpICU+JQ0KICBtdXRhdGUoYWNyb3NzKHdoZXJlKGlzLm51bWVyaWMpLCBleHApKSAlPiUNCiAgbXV0YXRlKGFjcm9zcyh3aGVyZShpcy5udW1lcmljKSwgfiByb3VuZCguLCAyKSkpICU+JQ0KICBndCgpDQpgYGANCg0KUHVsbGluZyB0aGUgZW1tZWFucyBkcmF3cyBmb3Igb3VyIHBsb3QNCg0KYGBge3J9DQplbW1lYW5zX2RyYXdzX3JtciA8LSBtbzJfbW9kX2dhbW1hICU+JQ0KICBlbW1lYW5zKH4gc2FsaW5pdHlfZ3JvdXApICU+JQ0KICBnYXRoZXJfZW1tZWFuc19kcmF3cygpICU+JSANCiAgZHBseXI6Om11dGF0ZSgudmFsdWUgPSAgZXhwKC52YWx1ZSksDQogICAgICAgICAgICAgICAgc2FsaW5pdHlfZ3JvdXAgPSBhcy5jaGFyYWN0ZXIoc2FsaW5pdHlfZ3JvdXApKQ0KDQplbW1lYW5zX2NvbnRyYXN0X2RyYXdzX3JtciAgPC0gIG1vMl9tb2RfZ2FtbWEgJT4lDQogIGVtbWVhbnMofiBzYWxpbml0eV9ncm91cCkgJT4lDQogIGNvbnRyYXN0KG1ldGhvZCA9ICJwYWlyd2lzZSIpICU+JQ0KICBnYXRoZXJfZW1tZWFuc19kcmF3cygpICU+JSANCiAgZHBseXI6Om11dGF0ZSgudmFsdWUgPSAgZXhwKC52YWx1ZSkpDQpgYGANCg0KIyMjIyBGaWd1cmUgUzcgKEZpZ3VyZSAxYSkNCg0KKioqTk9URTogVGhpcyBwbG90IGlzIGluIHRoZSBtYWluIHRleHQgb2YgdGhlIG1hbnVzY3JpcHQgYXMgRmlndXJlIDFhKioqDQoNCmBgYHtyfQ0KZmlnXzFhIDwtIGdncGxvdCgpICsNCiAgZ2VvbV92aW9saW4oZGF0YSA9IHNsb3BlX3RpZHlfc21yLA0KICAgICAgICAgICAgICBhZXMoeCA9IHNhbGluaXR5X2dyb3VwLCB5ID0gTU8yLCBmaWxsID0gc2FsaW5pdHlfZ3JvdXApLA0KICAgICAgICAgICAgICBjb2xvciA9IE5BLCBhbHBoYSA9IDAuMikgKw0KICBnZW9tX2ppdHRlcihkYXRhID0gc2xvcGVfdGlkeV9zbXIsDQogICAgICAgICAgICAgIGFlcyh4ID0gc2FsaW5pdHlfZ3JvdXAsIHkgPSBNTzIsIGZpbGwgPSBzYWxpbml0eV9ncm91cCksDQogICAgICAgICAgICAgIHNoYXBlID0gMjEsIHdpZHRoID0gMC4zLCBzaXplID0gMSwgY29sb3IgPSAiYmxhY2siLCBhbHBoYSA9IDAuMSkgKw0KICBnZW9tX3BvaW50KGRhdGEgPSBtZWFuX21vMl9zYWxpbml0eSwNCiAgICAgICAgICAgICBhZXMoeCA9IHNhbGluaXR5X2dyb3VwLCB5ID0gbWVhbl9tbzIsIGZpbGwgPSBzYWxpbml0eV9ncm91cCksDQogICAgICAgICAgICAgc2l6ZSA9IDQsIGFscGhhID0gMSwgc3Ryb2tlID0gMiwgY29sb3IgPSAiYmxhY2siLCBzaGFwZSA9IDIxLA0KICAgICAgICAgICAgIHBvc2l0aW9uID0gcG9zaXRpb25fbnVkZ2UoeCA9IDAuMDUpKSArDQogIHN0YXRfcG9pbnRpbnRlcnZhbChkYXRhID0gZW1tZWFuc19kcmF3c19ybXIsIA0KICAgICAgICAgICAgICAgICAgICAgYWVzKHggPSBzYWxpbml0eV9ncm91cCwgeSA9IC52YWx1ZSksDQogICAgICAgICAgICAgICAgICAgICBjb2xvciA9ICJibGFjayIsIGZpbGwgPSAiZ3JleSIsIHBvaW50X2ludGVydmFsID0gIm1lYW5fcWkiLCAud2lkdGggPSAwLjk1LCBzaGFwZSA9IDIxLCAgc3Ryb2tlID0gMiwgcG9pbnRfc2l6ZSA9IDQsIGFscGhhID0gMSwNCiAgICAgICAgICAgICAgICAgICAgIHBvc2l0aW9uID0gcG9zaXRpb25fbnVkZ2UoeCA9IC0wLjA1KSkgKw0KICBzY2FsZV95X2NvbnRpbnVvdXMobGltaXRzID0gYygwLjAwLCAwLjM1KSkgKyAgIyDwn5GIIFNldCB5LWF4aXMgcmFuZ2UgaGVyZQ0KICBzY2FsZV9maWxsX21hbnVhbCh2YWx1ZXMgPSBjKCIjNEI1MzIwIiwgIiMwMDAwODAiKSkgKw0KICBzY2FsZV9jb2xvdXJfbWFudWFsKHZhbHVlcyA9IGMoIiM0QjUzMjAiLCAiIzAwMDA4MCIpKSArDQogIHRoZW1lKA0KICAgIGxlZ2VuZC5wb3NpdGlvbiA9ICJub25lIiwNCiAgICBwYW5lbC5ncmlkLm1ham9yID0gZWxlbWVudF9ibGFuaygpLA0KICAgIHBhbmVsLmdyaWQubWlub3IgPSBlbGVtZW50X2JsYW5rKCksDQogICAgcGFuZWwuYm9yZGVyID0gZWxlbWVudF9yZWN0KGNvbG91ciA9ICJibGFjayIsIGZpbGwgPSBOQSwgbGluZXdpZHRoID0gMSksDQogICAgcGFuZWwuYmFja2dyb3VuZCA9IGVsZW1lbnRfcmVjdChmaWxsID0gIndoaXRlIiwgY29sb3VyID0gTkEpLA0KICAgIHBsb3QuYmFja2dyb3VuZCA9IGVsZW1lbnRfcmVjdChmaWxsID0gIndoaXRlIiwgY29sb3VyID0gTkEpDQogICkgKw0KICBsYWJzKA0KICAgIHN1YnRpdGxlID0gIiIsDQogICAgeCA9ICJTYWxpbml0eSBncm91cCAocHB0KSIsDQogICAgeSA9ICJSb3V0aW5lIE1PMiAobWcgTzIgZy9oKSINCiAgKQ0KDQpmaWdfMWENCmBgYA0KDQoqKkZpZ3VyZSAxYToqKiBSb3V0aW5lIG1ldGFib2xpYyByYXRlIChpLmUuIE1PPHN1Yj4yPC9zdWI+IChtZ14tMV4gTzxzdWI+Mjwvc3ViPiBoXi0xXikgbWVhc3VyZWQgZHVyaW5nIFNNUiBtZWFzc3VybWVudHMpIHBsb3R0ZWQgYnkgc2FsaW5pdHkgdHJlYXRtZW50LiBUaGUgc21hbGwgdHJhbnNwYXJlbnQgcG9pbnRzIGFyZSB0aGUgb2JzZXJ2ZWQgdmFsdWVzLCB0aGUgc2hhZGVkIGFyZWEgYmVoaW5kIHRoZSBwb2ludHMgaXMgdGhlIGEga2VybmVsIGRlbnNpdHkgb2YgdGhlIG9ic2VydmVkIGRhdGEsIHRoZSBsYXJnZSBjb2xvdXJlZCBwb2ludCAodG8gdGhlIHJpZ2h0KSBpcyB0aGUgb2JzZXJ2ZWQgbWVhbiwgdGhlIGxhcmdlIGdyZXkgcG9pbnQgd2l0aCBlcnJvciBiYXJzICh0byB0aGUgbGVmdCkgaXMgdGhlIG1vZGVsIGVzdGltYXRlZCBtYXJnaW5hbCBtZWFuIChlZW1lYW4pIDk1JSBDcmVkaWJsZSBJbnRlcnZhbHMgKDk1JSBDSSkuDQoNCmBgYHtyfQ0KIyBnZ3NhdmUoZmlsZW5hbWUgPSBwYXN0ZTAob3V0cHV0X2ZpZ193ZCwgIi4vbXNfZmlndXJlcy4vZmlndXJlLTFhLnBkZiIpLA0KIyAgICAgICAgcGxvdCA9IGZpZ18xYSwNCiMgICAgICAgIHdpZHRoID0gMjEwLzIsDQojICAgICAgICBoZWlnaHQgPSAyOTcvMiwgICMgU3BlY2lmeSB0aGUgaGVpZ2h0IG9mIHRoZSBwbG90DQojICAgICAgICB1bml0cyA9ICJtbSIpDQpgYGANCg0KIyMgU01SDQoNCiMjIyBGb3JtYXRpbmcgYW5kIHNjYWxpbmcgZGF0YQ0KDQpIZXJlIHdlIGFyZSBmaWx0ZXJpbmcgdGhlIGRhdGEgZnJhbWUgdG8gaGF2ZSBvbmx5IG1lYXN1cmUgcGVyIGZpc2ggZm9yIHRoZSBTTVIgZXN0aW1hdGUNCg0KYGBge3J9DQpzY2FsZV9saXN0IDwtIGMoInRlbXBfbWVhbiIsICJsb2dfbWFzcyIsICJjeWNsZXMiKQ0KDQpzbXIgPC0gc2xvcGVfdGlkeV9zbXIgJT4lDQogIGRwbHlyOjpncm91cF9ieShpZCkgJT4lIA0KICBkcGx5cjo6cmVmcmFtZSh0ZW1wX21lYW4gPSBtZWFuKHRlbXApLA0KICAgICAgICAgICAgICAgICBsb2dfbWFzcyA9IGxvZ19tYXNzWzFdLA0KICAgICAgICAgICAgICAgICBtYXNzID0gbWFzc1sxXSwNCiAgICAgICAgICAgICAgICAgU01SID0gU01SWzFdLA0KICAgICAgICAgICAgICAgICBzYWxpbml0eV9ncm91cCA9IHNhbGluaXR5X2dyb3VwWzFdLA0KICAgICAgICAgICAgICAgICBjeWNsZXMgPSBsZW5ndGgob3JkZXIpKSAlPiUgDQogIGRwbHlyOjptdXRhdGUoYWNyb3NzKGFsbF9vZihzY2FsZV9saXN0KSwgfiBzY2FsZSgueCwgY2VudGVyID0gVFJVRSwgc2NhbGUgPSBGQUxTRSksDQogICAgICAgICAgICAgICAgICAgICAgIC5uYW1lcyA9ICJ7LmNvbH1fYyIpKQ0KYGBgDQoNClN1bW1hcnkgb2YgcHJlZGljdG9ycyB1c2VkIGluIHRoZSBtb2RlbCBiZWxvdw0KDQpgYGB7cn0NCnNtciAlPiUNCiAgZHBseXI6OnJlZnJhbWUodGVtcF9yYW5nZSA9IHBhc3RlMChyb3VuZChtaW4odGVtcF9tZWFuLCBuYS5ybSA9IFRSVUUpLDIpLCAi4oCTIiwgcm91bmQobWF4KHRlbXBfbWVhbiwgbmEucm0gPSBUUlVFKSwyKSksDQogICAgICAgICAgICBtYXNzX3JhbmdlID0gcGFzdGUwKHJvdW5kKG1pbihtYXNzLCBuYS5ybSA9IFRSVUUpLDIpLCAi4oCTIiwgcm91bmQobWF4KG1hc3MsIG5hLnJtID0gVFJVRSksMikpLA0KICAgICAgICAgIGN5Y2xlX3JhbmdlID0gcGFzdGUwKHJvdW5kKG1pbihjeWNsZXMsIG5hLnJtID0gVFJVRSksMiksICLigJMiLCByb3VuZChtYXgoY3ljbGVzLCBuYS5ybSA9IFRSVUUpLDIpKSwNCiAgICAgICAgICBudW1iZXJfZmlzaCA9IGxlbmd0aCh1bmlxdWUoaWQpKSAgDQogICAgICAgICAgKSAlPiUgDQogIGd0KCkNCmBgYA0KDQojIyMgTW9kZWwgc3RydWN0dXJlDQoNCkhlcmUgd2Ugd2lsbCB1c2UgYSBCYXllc2lhbiBHZW5lcmFsaXNlZCBMaW5lYXIgTWl4ZWQgTW9kZWwgKEdMTU0pIHdpdGggYSBHYW1tYSBkaXN0cmlidXRpb24gYW5kIGEgbG9nIGxpbmsgYEdhbW1hKGxpbmsgPSAibG9nIilgLCB3aGVyZSB0aGUgc2hhcGUgcGFyYW1ldGVyICjOsSkgaXMgYWxzbyBtb2RlbGxlZCBhcyBhIGZ1bmN0aW9uIG9mIHRoZSBzYWxpbml0eSBncm91cCwgYHNoYXBlIH4gc2FsaW5pdHlfZ3JvdXBgLiBUaGlzIG1vZGVscyBNTzIgYnkgc2FsaW5pdHkgZHVyaW5nIHRoZSBTTVIgcGhhc2UgdG8gc2VlIGlmIHRoZSBmaXNoIGhlbGQgYXQgZGlmZmVyZW50IHNhbGluaXRpZXMgaGF2ZSBkaWZmZXJlbnQgU01Scy4gV2UgaGF2ZSBhbHNvIGFkZGVkIGEgZmV3IHNjYWxlZCBwcmVkaWN0b3JzLCB0aGF0IG1heSBoZWxwIGRlc2NyaWJlIHZhcmlhdGlvbiBpbiB0aGUgZGF0YSwgc3VjaCBhcyBmaXNoIG1hc3MgKGc7IDAuMjEtLTEuNikgbWVhbiB0ZW1wZXJhdHVyZSAowrBDOyAxMy44NDEtLTE0LjI3NyksIGFuZCB0aGUgbnVtYmVyIG9mIGN5Y2xlcyBvdmVyIHdoaWNoIFNNUiB3YXMgZXN0aW1hdGVkICgxLS0yMykuIFdlIGFsbG93ZWQgdGhlIHRoZSBzaGFwZSBwYXJhbWV0ZXIgKM6xKSB0byB2YXJ5IGFzIGEgZnVuY3Rpb24gb2Ygc2FsaW5pdHlfZ3JvdXAgdG8gaW1wcm92ZSBmaXQuDQoNCmBgYHtyfQ0Kc21yX2dhbW1hX2JmIDwtIGJmKFNNUiB+IHRlbXBfbWVhbl9jICsgDQogICAgICAgICAgICAgICAgICBjeWNsZXNfYyAgKw0KICAgICAgICAgICAgICAgICAgbG9nX21hc3NfYyArIA0KICAgICAgICAgICAgICAgICAgc2FsaW5pdHlfZ3JvdXAsDQogICAgICAgICAgICAgICAgICBzaGFwZSB+IHNhbGluaXR5X2dyb3VwLA0KICAgICAgICAgICAgICAgICAgZmFtaWx5ID0gR2FtbWEobGluayA9ICJsb2ciKSkNCmBgYA0KDQojIyMgUHJpb3Igc2VsZWN0aW9uDQoNClRoZXNlIGFyZSB0aGUgZGVmYXVsdCBwcmlvcnMgZm9yIHRoZSBtb2RlbC4gV2Ugd2lsbCB1c2UgdGhlc2UuDQoNCmBgYHtyfQ0KcHJpb3JzX2RlZmF1bHQgPC0gZ2V0X3ByaW9yKHNtcl9nYW1tYV9iZiwgZGF0YSA9IHNtciwgZmFtaWx5ID0gR2FtbWEobGluayA9ICJsb2ciKSkNCnByaW9yc19kZWZhdWx0ICU+JSANCiAgZ3QoKQ0KYGBgDQoNCiMjIyBSdW4gbW9kZWwNCg0KSGVyZSB3ZSBydW4gdGhlIG1vZGVsLCBJIGhhdmUgaGFzaGVkIHRoaXMgb3V0IGJlY2F1c2UgSSBoYXZlIHNhdmVkIHRoZSBtb2RlbCBmb3IgcXVpY2sgcmVsb2FkaW5nLg0KDQrij63vuI8gKipTa2lwIHRoaXMgc3RlcCoqIGlmIHlvdSBoYXZlIGRvd25sb2FkZWQgdGhlIHNhdmVkIG1vZGVsICdzbXJfbW9kX2dhbW1hLnJkcycNCg0KYGBge3IsIGVjaG89RkFMU0V9DQojIHNtcl9tb2RfZ2FtbWFfYyA8LSBicm0oc21yX2dhbW1hX2JmLA0KIyAgICAgICAgICAgICAgICBkYXRhID0gc21yLA0KIyAgICAgICAgICAgICAgICBjb3JlcyA9IDQsDQojICAgICAgICAgICAgICAgIGNoYWlucyA9IDQsDQojICAgICAgICAgICAgICAgIHByaW9yID0gcHJpb3JzX2RlZmF1bHQsDQojICAgICAgICAgICAgICAgIHdhcm11cCA9IDEwMDAsDQojICAgICAgICAgICAgICAgIHNlZWQgPSAxNDMwMTksDQojICAgICAgICAgICAgICAgIHRoaW4gPSAyLA0KIyAgICAgICAgICAgICAgICBpdGVyID0gODAwMCwNCiMgICAgICAgICAgICAgICAgc2F2ZV9wYXJzID0gc2F2ZV9wYXJzKGFsbD1UUlVFKSwNCiMgICAgICAgICAgICAgICAgc2FtcGxlX3ByaW9yID0gVFJVRSwNCiMgICAgICAgICAgICAgICAgZmlsZSA9IHBhc3RlMChvdXRwdXRfbW9kc193ZCwgJy4vc21yX21vZF9nYW1tYScpKQ0KIyBwcmludCgiTW9kZWwgY29tcGxldGUiKQ0KYGBgDQoNCkhlcmUgd2UgcmVsb2FkIHRoZSBtb2RlbA0KDQpgYGB7cn0NCnNtcl9tb2RfZ2FtbWEgPC0gIHJlYWRSRFMoZmlsZSA9IHBhc3RlMChvdXRwdXRfbW9kc193ZCwgJy4vc21yX21vZF9nYW1tYS5yZHMnKSkNCmBgYA0KDQojIyMgTW9kZWwgZGlhZ25vc3RpY3MNCg0KQ2hlY2tpbmcgbW9kZWwgY29udmVyZ2VuY2UNCg0KYGBge3J9DQpjb2xvcl9zY2hlbWVfc2V0KCJyZWQiKQ0KcGxvdChzbXJfbW9kX2dhbW1hLCBhc2sgPSBGKQ0KYGBgDQoNClVzaW5nIGxlYXZlIG9uZSBvdXQgKGxvbykgbWVhc3VyZSBvZiBmaXQsIHRoZSBtb2RlbCBhcHBlYXJzIHRvIHByZWZvcm0gd2VsbCwgdHdvIFBhcmV0byBrIGVzdGltYXRlcyBmYWxscyBvdXRzaWRlIHRoZSBnb29kIHJhbmdlICgwLjcsIDFdDQoNCmBgYHtyfQ0KbG9vKHNtcl9tb2RfZ2FtbWEpDQpgYGANCg0KTW9kZWwgcHJlZGljdGlvbnMgZ2VuZXJhbGx5IGFsaWduIHdpdGggdGhlIG9ic2VydmVkIGRhdGEsIGJ1dCB0aGVyZSBpcyBhIGxvdCBvZiB1bmNlcnRhaW50eSBhcm91bmQgdGhpcyBlc3RpbWF0ZS4NCg0KYGBge3J9DQpjb2xvcl9zY2hlbWVfc2V0KCJyZWQiKQ0KcCA8LSBwcF9jaGVjayhzbXJfbW9kX2dhbW1hLCB0eXBlID0gImRlbnNfb3ZlcmxheSIpDQpwDQpgYGANCg0KIyMjIPCfk4ggUmVzdWx0cw0KDQpXZSBkaWQgbm90IHNlZSBhIG1lYW5pbmdmdWwgZGlmZmVyZW5jZSBiZXR3ZWVuIHRoZSByb3V0aW5lIG1ldGFib2xpYyByYXRlIGZvciBmaXNoIGZyb20gdGhlIHR3byBzYWxpbml0eSB0cmVhdG1lbnRzLg0KDQojIyMjIFRhYmxlIFMyDQoNCioqVGFibGUgUzI6KiogRml4ZWQgZWZmZWN0IEVzdGltYXRlcyAozrIpIGFuZCA5NSUgQ3JlZGlibGUgSW50ZXJ2YWxzICg5NSUgQ0kpIGZyb20gYSBCYXllc2lhbiBHZW5lcmFsaXNlZCBMaW5lYXIgTWl4ZWQgTW9kZWwgKEdMTU0pIHdpdGggYSBHYW1tYSBkaXN0cmlidXRpb24gYW5kIGEgbG9nIGxpbmsNCg0KYGBge3J9DQptb2RlbF9lc3QgPC0gZml4ZWYoc21yX21vZF9nYW1tYSwgcHJvYnMgPSBjKDAuMDI1LCAwLjk3NSkpICU+JSANCiAgYXMuZGF0YS5mcmFtZSgpICU+JSANCiAgdGliYmxlOjpyb3duYW1lc190b19jb2x1bW4odmFyID0gIlByZWRpY3RvciIpICU+JSANCiAgZHBseXI6Om11dGF0ZSgnzrInID0gcm91bmQoRXN0aW1hdGUsIDMpLA0KICAgICAgICAgICAgICAgIFEyLjUgPSByb3VuZChRMi41LCAzKSwNCiAgICAgICAgICAgICAgICBROTcuNSA9IHJvdW5kKFE5Ny41LCAzKSwNCiAgICAgICAgICAgICAgICAnOTUlIENJJyA9IHBhc3RlMCgiWyIsIFEyLjUsICIsICIsIFE5Ny41LCAiXSIpKQ0KDQptb2RlbF9lc3QgJT4lIA0KICBkcGx5cjo6c2VsZWN0KFByZWRpY3RvciwgJ86yJywgJzk1JSBDSScpICU+JSANCiAgZ3QoKQ0KYGBgDQoNCkxvb2tpbmcgYXQgdGhlIG1hcmdpbmFsIG1lYW4gZGlmZmVyZW5jZSBiZXR3ZWVuIHNhbGluaXR5IGdyb3Vwcw0KDQpgYGB7cn0NCmVtX3Jlc3VsdHMgPC0gZW1tZWFucyhzbXJfbW9kX2dhbW1hLCB+IHNhbGluaXR5X2dyb3VwKSANCg0KZW1fcmVzdWx0c19kZiA8LSAgZW1fcmVzdWx0cyAlPiUgDQogIGFzLmRhdGEuZnJhbWUoKSAlPiUNCiAgbXV0YXRlKGFjcm9zcyh3aGVyZShpcy5udW1lcmljKSwgfiByb3VuZChleHAoLiksMykpKQ0KDQoNCmVtX3Jlc3VsdHNfZGYgJT4lIA0KICBndCgpDQpgYGANCg0KYGBge3J9DQpjb250cmFzdF9yZXN1bHRzX2RmIDwtICBjb250cmFzdF9yZXN1bHRzICU+JSANCiAgYXMuZGF0YS5mcmFtZSgpICU+JQ0KICBtdXRhdGUoYWNyb3NzKHdoZXJlKGlzLm51bWVyaWMpLCB+IGV4cCguKSkpDQoNCmNvbnRyYXN0KGVtX3Jlc3VsdHMsIG1ldGhvZCA9ICJwYWlyd2lzZSIpICU+JSANCiAgYXMuZGF0YS5mcmFtZSgpICU+JQ0KICBtdXRhdGUoYWNyb3NzKHdoZXJlKGlzLm51bWVyaWMpLCB+IHJvdW5kKGV4cCguKSwyKSkpICU+JSANCiAgZ3QoKQ0KYGBgDQoNCkhlcmUgd2Ugd2UgY2FsY3VsYXRlIHRoZSBtb2RlbC1lc3RpbWF0ZWQgbWFyZ2luYWwgbWVhbnMgZm9yIFNNUiBmb3IgYSB1bml0IGluY3JlYXNlIGluIG1hc3MgYW5kIHRyYW5zZm9ybSB0aGlzIG9udG8gYSBmb2xkLWNoYW5nZSBzY2FsZS4gV2hlcmUgbG9nKG1hc3MpIGhhcyBiZSBjZW50ZXJlZCwgYSB2YXVsZSBvZiAwIHJlcHJlc2VudHMgdGhlIG1lYW4gbWFzcyBvZiBmaXNoLCBhbmQgMSBjb3JyZXNwb25kcyB0byBhIGZpc2ggd2l0aCBsb2cobWFzcykgPSAxIHVuaXQgYWJvdmUgdGhlIG1lYW4gKGkuZS4gMi43IHRpbWVzIGxhcmdlKS4NCg0KYGBge3J9DQplbV9tYXNzIDwtIGVtbWVhbnMoc21yX21vZF9nYW1tYSwgfiBsb2dfbWFzc19jLCBhdCA9IGxpc3QobG9nX21hc3NfYyA9IGMoMCwgMSkpKQ0KDQojIEdldCB0aGUgY29udHJhc3QgKGRpZmZlcmVuY2UgaW4gbG9nIE1P4oKCKSwgdGhlbiBiYWNrLXRyYW5zZm9ybQ0KY29udHJhc3QoZW1fbWFzcywgbWV0aG9kID0gInJldnBhaXJ3aXNlIikgJT4lDQogIGFzLmRhdGEuZnJhbWUoKSAlPiUNCiAgbXV0YXRlKGFjcm9zcyh3aGVyZShpcy5udW1lcmljKSwgZXhwKSkgJT4lDQogIG11dGF0ZShhY3Jvc3Mod2hlcmUoaXMubnVtZXJpYyksIH4gcm91bmQoLiwgMikpKSAlPiUNCiAgZ3QoKQ0KYGBgDQoNClB1bGxpbmcgdGhlIGVtbWVhbnMgZHJhd3MgZm9yIG91ciBwbG90DQoNCmBgYHtyfQ0KZW1tZWFuc19kcmF3X3NtciA8LSBzbXJfbW9kX2dhbW1hICU+JQ0KICBlbW1lYW5zKH4gc2FsaW5pdHlfZ3JvdXApICU+JQ0KICBnYXRoZXJfZW1tZWFuc19kcmF3cygpICU+JSANCiAgZHBseXI6Om11dGF0ZSgudmFsdWUgPSAgZXhwKC52YWx1ZSksDQogICAgICAgICAgICAgICAgc2FsaW5pdHlfZ3JvdXAgPSBhcy5jaGFyYWN0ZXIoc2FsaW5pdHlfZ3JvdXApKQ0KDQplbW1lYW5zX2NvbnRyYXN0X2RyYXdzX3NtciA8LSAgc21yX21vZF9nYW1tYSAlPiUNCiAgZW1tZWFucyh+IHNhbGluaXR5X2dyb3VwKSAlPiUNCiAgY29udHJhc3QobWV0aG9kID0gInBhaXJ3aXNlIikgJT4lDQogIGdhdGhlcl9lbW1lYW5zX2RyYXdzKCkgJT4lIA0KICBkcGx5cjo6bXV0YXRlKC52YWx1ZSA9ICBleHAoLnZhbHVlKSkNCmBgYA0KDQojIyMjIEZpZ3VyZSBTOCAoRmlndXJlIDFiKQ0KDQpUaGlzIHBsb3QgaXMgaW4gdGhlIG1haW4gdGV4dCBvZiB0aGUgbWFudXNjcmlwdCBhcyAqKkZpZ3VyZSAxYioqDQoNCmBgYHtyfQ0KbWVhbl9zbXJfc2FsaW5pdHkgPC0gc21yICU+JQ0KICBkcGx5cjo6Z3JvdXBfYnkoc2FsaW5pdHlfZ3JvdXApICU+JQ0KICBkcGx5cjo6cmVmcmFtZShtZWFuX1NNUiA9IG1lYW4oU01SLCBuYS5ybSA9IFRSVUUpKQ0KDQoNCmZpZ18xYiA8LSBnZ3Bsb3QoKSArDQogIGdlb21fdmlvbGluKGRhdGEgPSBzbXIsDQogICAgICAgICAgICAgIGFlcyh4ID0gc2FsaW5pdHlfZ3JvdXAsIHkgPSBTTVIsIGZpbGwgPSBzYWxpbml0eV9ncm91cCksDQogICAgICAgICAgICAgIGNvbG9yID0gTkEsIGFscGhhID0gMC4yKSArDQogIGdlb21faml0dGVyKGRhdGEgPSBzbXIsDQogICAgICAgICAgICAgIGFlcyh4ID0gc2FsaW5pdHlfZ3JvdXAsIHkgPSBTTVIsIGZpbGwgPSBzYWxpbml0eV9ncm91cCksDQogICAgICAgICAgICAgIHNoYXBlID0gMjEsIHdpZHRoID0gMC4zLCBzaXplID0gMSwgY29sb3IgPSAiYmxhY2siLCBhbHBoYSA9IDAuMSkgKw0KICBnZW9tX3BvaW50KGRhdGEgPSBtZWFuX3Ntcl9zYWxpbml0eSwNCiAgICAgICAgICAgICBhZXMoeCA9IHNhbGluaXR5X2dyb3VwLCB5ID0gbWVhbl9TTVIsIGZpbGwgPSBzYWxpbml0eV9ncm91cCksDQogICAgICAgICAgICAgc2l6ZSA9IDQsIGFscGhhID0gMSwgc3Ryb2tlID0gMiwgY29sb3IgPSAiYmxhY2siLCBzaGFwZSA9IDIxLA0KICAgICAgICAgICAgIHBvc2l0aW9uID0gcG9zaXRpb25fbnVkZ2UoeCA9IDAuMDUpKSArDQogIHN0YXRfcG9pbnRpbnRlcnZhbChkYXRhID0gZW1tZWFuc19kcmF3X3NtciwNCiAgICAgICAgICAgICAgICAgICAgIGFlcyh4ID0gc2FsaW5pdHlfZ3JvdXAsIHkgPSAudmFsdWUpLA0KICAgICAgICAgICAgICAgICAgICAgY29sb3IgPSAiYmxhY2siLCBmaWxsID0gImdyZXkiLCBwb2ludF9pbnRlcnZhbCA9ICJtZWFuX3FpIiwgLndpZHRoID0gMC45NSwgc2hhcGUgPSAyMSwgIHN0cm9rZSA9IDIsIHBvaW50X3NpemUgPSA0LCBhbHBoYSA9IDEsDQogICAgICAgICAgICAgICAgICAgICBwb3NpdGlvbiA9IHBvc2l0aW9uX251ZGdlKHggPSAtMC4wNSkpICsNCiAgc2NhbGVfeV9jb250aW51b3VzKGxpbWl0cyA9IGMoMC4wMCwgMC4zNSkpICsNCiAgc2NhbGVfZmlsbF9tYW51YWwodmFsdWVzID0gYygiIzRCNTMyMCIsICIjMDAwMDgwIikpICsgICMgQ3VzdG9tIGZpbGwgY29sb3Vycw0KICBzY2FsZV9jb2xvdXJfbWFudWFsKHZhbHVlcyA9IGMoIiM0QjUzMjAiLCAiIzAwMDA4MCIpKSArDQogIHRoZW1lKA0KICAgIGxlZ2VuZC5wb3NpdGlvbiA9ICJub25lIiwNCiAgICBwYW5lbC5ncmlkLm1ham9yID0gZWxlbWVudF9ibGFuaygpLA0KICAgIHBhbmVsLmdyaWQubWlub3IgPSBlbGVtZW50X2JsYW5rKCksDQogICAgcGFuZWwuYm9yZGVyID0gZWxlbWVudF9yZWN0KGNvbG91ciA9ICJibGFjayIsIGZpbGwgPSBOQSwgbGluZXdpZHRoID0gMSksDQogICAgcGFuZWwuYmFja2dyb3VuZCA9IGVsZW1lbnRfcmVjdChmaWxsID0gIndoaXRlIiwgY29sb3VyID0gTkEpLA0KICAgIHBsb3QuYmFja2dyb3VuZCA9IGVsZW1lbnRfcmVjdChmaWxsID0gIndoaXRlIiwgY29sb3VyID0gTkEpDQogICkgKw0KICBsYWJzKA0KICAgIHN1YnRpdGxlID0gIiIsDQogICAgeCA9ICJTYWxpbml0eSBncm91cCAocHB0KSIsDQogICAgeSA9ICJTdGFuZGFyZCBtZXRhYm9saWMgcmF0ZSAoU01SOyBtZyBPMiBnL2gpIg0KICApDQoNCmZpZ18xYg0KYGBgDQoNCmxhYnMoDQogICAgeSA9IGV4cHJlc3Npb24oIm1nIE8iWzJdfiJoIl4tMSksDQogICAgeCA9ICJEaXNzb2x2ZWQgb3h5Z2VuIChETzsgJSBzYXR1cmF0aW9uKSINCiAgKSArDQoqKkZpZ3VyZSAxYjoqKiBUaGUgc3RhbmRhcmQgbWV0YWJvbGljIHJhdGUgZXN0aW1hdGUgKG1nXi0xXiBPPHN1Yj4yPC9zdWI+IGheLTFeKSBwbG90dGVkIGJ5IHNhbGluaXR5IHRyZWF0bWVudC4gVGhlIHNtYWxsIHRyYW5zcGFyZW50IHBvaW50cyBhcmUgdGhlIG9ic2VydmVkIHZhbHVlcywgdGhlIHNoYWRlZCBhcmVhIGJlaGluZCB0aGUgcG9pbnRzIGlzIHRoZSBhIGtlcm5lbCBkZW5zaXR5IG9mIHRoZSBvYnNlcnZlZCBkYXRhLCB0aGUgbGFyZ2UgY29sb3VyZWQgcG9pbnQgKHRvIHRoZSByaWdodCkgaXMgdGhlIG9ic2VydmVkIG1lYW4sIHRoZSBsYXJnZSBncmV5IHBvaW50IHdpdGggZXJyb3IgYmFycyAodG8gdGhlIGxlZnQpIGlzIHRoZSBtb2RlbCBlc3RpbWF0ZWQgbWFyZ2luYWwgbWVhbiAoZWVtZWFuKSA5NSUgQ3JlZGlibGUgSW50ZXJ2YWxzICg5NSUgQ0kpLg0KDQpgYGB7cn0NCiMgZ2dzYXZlKGZpbGVuYW1lID0gcGFzdGUwKG91dHB1dF9maWdfd2QsICIuL21zX2ZpZ3VyZXMuL2ZpZ3VyZS0xYi5wZGYiKSwNCiMgICAgICAgIHBsb3QgPSBmaWdfMWIsDQojICAgICAgICB3aWR0aCA9IDIxMC8yLA0KIyAgICAgICAgaGVpZ2h0ID0gMjk3LzIsICAjIFNwZWNpZnkgdGhlIGhlaWdodCBvZiB0aGUgcGxvdA0KIyAgICAgICAgdW5pdHMgPSAibW0iKQ0KYGBgDQoNCiMjIEluY3JlbWVudGFsIHJlZ3Jlc3Npb24gYW5hbHlzZXMNCg0KSGVyZSB3ZSBhcmUgZm9sbG93aW5nIHRoZSBtZXRob2RzIFVyYmluYSBldCBhbC4gKDIwMTIpXlsxXV4gd2l0aCBhbiBpbmNyZW1lbnRhbCByZWdyZXNzaW9uIGFuYWx5c2VzLCBpbiBvcmRlciB0byBkZXRlcm1pbmUgdGhlIGJlc3QgZml0IGZvciB0aGUgTU88c3ViPjI8L3N1Yj4gdnMgTzxzdWI+Mjwvc3ViPiBkYXRhDQoNClRoaXMgYW5hbHlzaXMgYXBwcm9hY2ggZXZhbHVhdGVzIHRoZSByZWxhdGl2ZSAnZml0JyBvZiBlYWNoIHBvbHlub21pYWwgb3JkZXIgZXF1YXRpb24gc3RhcnRpbmcgYXQgemVybyBhbmQgaW5jcmVhc2luZyB0byB0aGUgdGhpcmQgb3JkZXIsIHBlcm1pdHRpbmcgYSBtYXRoZW1hdGljYWwgYXNzZXNzbWVudCBvZiB3aGV0aGVyIGZpc2ggd2VyZSBveHljb25mb3JtaW5nIG9yIG94eXJlZ3VsYXRvcmluZy4gSWYgdGhlIGRhdGEgaXMgYmVzdCBmaXR0ZWQvcHJlZGljdGVkIGJ5IGEgc2luZ2xlIGxpbmVhciByZWxhdGlvbnNoaXAgKDFec3ReLW9yZGVyIHBvbHlub21pYWwpIHdpdGggYSBwb3NpdGl2ZSBzbG9wZSwgdGhpcyB3b3VsZCBzdWdnZXN0cyB0aGUgZmlzaCB3ZXJlIG94eWNvbmZvcm1pbmcuIEFsdGVybmF0ZWx5LCBpZiB0aGUgcmVsYXRpb25zaGlwIGlzIGJlc3QgbW9kZWxsZWQgYnkgYSBmbGF0IHJlZ3Jlc3Npb24gKDBedGheLW9yZGVyIHBvbHlub21pYWwpLCBvciBhIGhpZ2hlciBvcmRlciBwb2x5bm9taWFsICgyXm5kXiBvciAzXnJkXi1vcmRlciBwb2x5bm9taWFsKSB0aGUgZmlzaCBpcyBsaWtlbHkgb3h5cmVndWxhdG9yaW5nLg0KDQojIyMgQnVpbGRpbmcgQmF5ZXNpYW4gcmVncmVzc2lvbnMNCg0KSGVyZSB3ZSBhcmUgdXNpbmcgYSBCYXllc2lhbiBhcHByb2FjaCB0byBtb2RlbCBmaXR0aW5nIHdpdGggYnJtLiBUaGVzZSBtb2RlbHMgdGFrZSBhIGxvbmcgdGltZSB0byBydW4sIHNvIEkgaGF2ZSBzYXZlZCB0aGVtIGFuZCByZS1sb2FkZWQgdGhlbSB0byBzYXZlIHRpbWUuIEkgaGF2ZSBhbHNvIHNhdmVkIHRoZSBzdW1tYXJ5IGRhdGEgcHJvZHVjZWQgZnJvbSB0aGUgbW9kZWxzLCB0byBzYXZlIHRpbWUsIHlvdSBjYW4gc2ltcGx5IHNraXAgdGhlIGhhc2hlZCBjb2RlIGFuZCBpbnB1dCB0aGUgcmVzdWx0aW5nIHN1bW1hcnkgZGF0YS4NCg0KV2Ugd2lsbCBydW4gb3VyIGN1c3RvbSBmdW5jdGlvbiwgYGJheWVzX2luY3JlbWVudGFsX3JlZ3Jlc3Npb25fYnlfaWRgLg0KDQpMZXQncyBtYWtlIGRlZmluZSB0aGUgb3V0cHV0IGRpcmVjdG9yeQ0KDQpgYGB7cn0NCm91dHB1dF9tb2RzX2JheWVzX3dkIDwtIHBhc3RlMCh3ZCwgIi4vb3V0cHV0LW1vZC4vYmF5ZXMtcmVncyIpDQpgYGANCg0K4o+t77iPICoqU2tpcCB0aGlzIHN0ZXAqKiBpZiB5b3UgaGF2ZSBhbHJlYWR5IHJ1biB0aGlzIG9uY2UsIG9yIGhhdmUgZG93bmxvYWRlZCB0aGUgc2F2ZWQgbW9kZWxzIG9yIHNhdmVkIGRhdGEgZmlsZXMgZnJvbSBHaXRIdWIgKHRoYXQncyB3aHkgaXRzIGhhc2hlZCBvdXQpLiDijJsqVGhpcyBjb2RlIHRha2VzIGEgd2hpbGUgdG8gcnVuICrijJsNCg0KRHVyaW5nIHRoaXMgcGFydCBvZiB0aGUgc2NyaXB0IEkgcmVjZWl2ZWQgYW4gZXJyb3IsICdDYXVzZWQgYnkgZXJyb3IgaW4gYHNvY2tldENvbm5lY3Rpb24oKWA6Jy4gSSB0aGluayB0aGUgc3lzdGVtIG1heSBiZSBoaXR0aW5nIGEgbGltaXQgb24gdGhlIG51bWJlciBvZiBzaW11bHRhbmVvdXMgc29ja2V0IGNvbm5lY3Rpb25zLiBJZiB5b3UgdGhpcyBwYXJ0IG9mIHRoZSBjb2RlIGFuZCBhbHNvIGdldCB0aGlzIGlzc3VlLCB0cnkgcmVkdWNpbmcgdGhlIG51bWJlciBvZiBwYXJhbGxlbCB3b3JrZXJzOiBwbGFuKG11bHRpc2Vzc2lvbiwgd29ya2VycyA9IDIpLCBvciBydW4gYWdhaW4gYWZ0ZXIgdGhlIGVycm9yLg0KDQpgYGB7ciwgcmVzdWx0cz0naGlkZSd9DQojIGlkcyA8LSBzbG9wZV90aWR5ICU+JQ0KIyAgIGRwbHlyOjpkaXN0aW5jdChpZCkgJT4lDQojICAgcHVsbChpZCkNCiMgDQojIHBsYW4obXVsdGlzZXNzaW9uKQ0KIyANCiMgZnV0dXJlX21hcCgNCiMgICBpZHMsDQojICAgYmF5ZXNfaW5jcmVtZW50YWxfcmVncmVzc2lvbl9ieV9pZCwNCiMgICBpZF9uYW1lID0gImlkIiwNCiMgICBkYXRhID0gc2xvcGVfdGlkeSwNCiMgICByZXNwb25zZSA9ICJNTzJfZyIsDQojICAgcHJlZGljdG9yID0gIkRPIiwNCiMgICBzZWVkX251bWJlciA9IDE0MzAxOSwNCiMgICBzYXZlX21vZGVscyA9IFRSVUUsDQojICAgbW9kX291dHB1dF93ZCA9IG91dHB1dF9tb2RzX2JheWVzX3dkDQojICkNCiMgDQojIHBsYW4oc2VxdWVudGlhbCkNCmBgYA0KDQpMb2FkIGFsbCBtb2RlbHMgYW5kIHN0b3JlIGluIGEgbGlzdCwgd2lsbCB1c2UgYSBsb3Qgb2YgbWVtb3J5LiBZb3UgY2FuIGFsc28gc2tpcCB0aGlzIHN0ZXAgYW5kIGxvYWQgdGhlIHJlc3VsdGluZyBkYXRhIGZyYW1lcyBiZWxvdy4gSSBhbSB1c2luZyB0aGUgY3VzdG9tIGZ1bmN0aW9uIGBsb2FkX3Jkc2AsIHNvIHdlIGNhbiBjb21wYXJlIHRoZW0gYW5kIGdlbmVyYXRlIHByZWRpY3Rpb25zLg0KDQrij63vuI8gKipTa2lwIHRoaXMgc3RlcCoqIGlmIHlvdSBoYXZlIGRvd25sb2FkZWQgdGhlIHNhdmVkIGRhdGEgcHVsbGVkIGZyb20gdGhlc2UgbW9kZWxzIGhhdmUgYWxzbyBoYXNlZCB0aGlzIG91dCwgYmF5ZXNfcmVnX21vZHNfZml0LmNzdicgYW5kICdiYXllc19yZWdfbW9kc19wcmVkaWN0aW9ucy5jc3YnLiBUaGVzZSBhcmUgaW4gdGhlIG1vZC1kYXRhIGRpcmVjdG9yeS4NCg0KYGBge3J9DQojIGJheWVzX3JlZ19tb2RzIDwtIGxvYWRfcmRzKG1vZGVsX2R3ID0gb3V0cHV0X21vZHNfYmF5ZXNfd2QpDQpgYGANCg0KIyMjIE1vZGVsIGZpdHMNCg0K4o+t77iPICoqU2tpcCB0aGlzIHN0ZXAqKiBpZiB5b3UgaGF2ZSBkb3dubG9hZGVkICoq8J+SvyBiYXllc19yZWdfbW9kc19maXQuY3N2KiouDQoNCkl0IGdldHMgbW9kZWwgZml0IHBhcmFtZXRlcnMgbG9vIGFuZCByMiB1c2luZyB0aGUgY3VzdG9tIGZ1bmN0aW9uLCBgaW5jcmVtZW50YWxfcmVncmVzc2lvbl9iYXllc19maXRzYC4gKuKMmyBUaGlzIGNvZGUgdGFrZXMgYSB3aGlsZSB0byBydW4g4oybKg0KDQpgYGB7cn0NCiMgYmF5ZXNfcmVnX21vZHNfZml0IDwtIGluY3JlbWVudGFsX3JlZ3Jlc3Npb25fYmF5ZXNfZml0cyhtb2RlbHMgPSBiYXllc19yZWdfbW9kcykNCiMgd3JpdGUuY3N2KGJheWVzX3JlZ19tb2RzX2ZpdCwgcGFzdGUwKG1vZF9kYXRhX3dkLCAiLi9iYXllc19yZWdfbW9kc19maXQuY3N2IiksIHJvdy5uYW1lcyA9IEZBTFNFKQ0KYGBgDQoNClJlYWRpbmcgaW4gdGhpcyBtb2RlbCBmaXQgZGF0YSBmcmFtZSwgaW4gdGhlIGNhc2UgeW91IGRpZCBub3QgbG9hZCBpbiBhbGwgdGhlIG1vZGVscy4NCg0KYGBge3J9DQpiYXllc19yZWdfbW9kc19maXQgPC0gcmVhZC5jc3YocGFzdGUwKG1vZF9kYXRhX3dkLCAiLi9iYXllc19yZWdfbW9kc19maXQuY3N2IikpDQpgYGANCg0KKipTZWxlY3RpbmcgdGhlIGJlc3QgZml0dGluZyBtb2RlbCoqDQoNCmVscGRfbG9vLCBvciB0aGUgZXhwZWN0ZWQgbG9nIHBvaW50d2lzZSBwcmVkaWN0aXZlIGRlbnNpdHkgZm9yIGxlYXZlLW9uZS1vdXQgY3Jvc3MtdmFsaWRhdGlvbiwgaXMgYSBtZXRyaWMgdXNlZCBpbiBCYXllc2lhbiBtb2RlbCBldmFsdWF0aW9uIHRvIGFzc2VzcyB0aGUgcHJlZGljdGl2ZSBhY2N1cmFjeSBvZiBhIG1vZGVsLiBUaGUgZWxwZF9sb28gaXMgYW4gYXBwcm94aW1hdGlvbiBvZiBob3cgd2VsbCB0aGUgbW9kZWwgaXMgZXhwZWN0ZWQgdG8gcHJlZGljdCBuZXcgZGF0YSwgYmFzZWQgb24gbGVhdmUtb25lLW91dCBjcm9zcy12YWxpZGF0aW9uLiBIaWdoZXIgZWxwZF9sb28gdmFsdWVzIGluZGljYXRlIGJldHRlciBwcmVkaWN0aXZlIHBlcmZvcm1hbmNlLg0KDQpgYGB7cn0NCmJlc3RfZml0X2JheWVzX3JlZyA8LSBiYXllc19yZWdfbW9kc19maXQgJT4lDQogIGRwbHlyOjpncm91cF9ieShpZCkgJT4lDQogIGRwbHlyOjptdXRhdGUoZWxwZF9sb29fcmFuayA9IHJhbmsoLWVscGRfbG9vKSkgJT4lDQogIGRwbHlyOjpzZWxlY3QoaWQsIG1vZGVsX3R5cGUsIGVscGRfbG9vLCByMiwgZWxwZF9sb29fcmFuaywgcjJfcTIuNSwgcjJfcTk3LjUsIGVzdGltYXRlX0RPLCBjb25mLmxvd19ETywgY29uZi5oaWdoX0RPKSAlPiUNCiAgdW5ncm91cCgpDQpgYGANCg0KIyMjIE1vZGVsIHByZWRpY3Rpb25zDQoNCuKPre+4jyAqKlNraXAgdGhpcyBzdGVwKiogaWYgeW91IGhhdmUgZG93bmxvYWRlZCAqKvCfkr8gYmF5ZXNfcmVnX21vZHNfcHJlZGljdGlvbnMuY3N2KiouDQoNCkl0IHB1bGxzIG91ciBtb2RlbCBwcmVkaWN0aW9ucyB1c2luZyBhIGN1c3RvbSBmdW5jdGlvbiBgYmF5ZXNfbW9kX3ByZWRpY3Rpb25zYC4NCg0KYGBge3J9DQojIGJheWVzX3JlZ19tb2RzX3ByZWRpY3Rpb25zIDwtIGJheWVzX21vZF9wcmVkaWN0aW9ucyhtb2RlbHMgPSBiYXllc19yZWdfbW9kcywgb3JpZ2luYWxfZGF0YSA9IHNsb3BlX3RpZHkpDQojIHdyaXRlLmNzdihiYXllc19yZWdfbW9kc19wcmVkaWN0aW9ucywgcGFzdGUwKG1vZF9kYXRhX3dkLCAiLi9iYXllc19yZWdfbW9kc19wcmVkaWN0aW9ucy5jc3YiKSwgcm93Lm5hbWVzID0gRkFMU0UpDQpgYGANCg0KUmVhZGluZyBpbiB0aGUgcHJlZGljdGVkIGRhdGENCg0KYGBge3J9DQpiYXllc19yZWdfbW9kc19wcmVkaWN0aW9ucyA8LSByZWFkLmNzdihwYXN0ZTAobW9kX2RhdGFfd2QsICIuL2JheWVzX3JlZ19tb2RzX3ByZWRpY3Rpb25zLmNzdiIpKQ0KYGBgDQoNCldlIGFyZSBnb2luZyB0byBjb21iaW5lZCB0aGlzIHdpdGggb3VyIGJlc3QgZml0dGluZyBtb2RlbCBkZiwgc28gd2Uga25vdyBob3cgdGhleSByYW5rcyBmb3IgTE9PLg0KDQpgYGB7cn0NCmJheWVzX3JlZ19tb2RzX3ByZWRpY3Rpb25zIDwtIGxlZnRfam9pbihiYXllc19yZWdfbW9kc19wcmVkaWN0aW9ucywgYmVzdF9maXRfYmF5ZXNfcmVnLCBieSA9IGMoImlkIiwgIm1vZGVsX3R5cGUiKSkNCmBgYA0KDQojIyMg8J+TiCBSZXN1bHRzDQoNCiMjIyMgTW9kZWwgc2VsZWN0aW9uIHN1bW1hcnkNCg0KSW4gbW9zdCBjYXNlcywgdGhlIHJlbGF0aW9uc2hpcCBiZXR3ZWVuIE1PPHN1Yj4yPC9zdWI+IGFuZCBETyB3YXMgYmVzdCBtb2RlbGxlZCB3aXRoIGEgMl5uZF4tb3JkZXIgcG9seW5vbWlhbCAoKm4qID0gMjIsIDM4JSBvZiBmaXNoKSwgZm9sbG93ZWQgYnkgYSAzXnJkXi1vcmRlciBwb2x5bm9taWFsICgqbiogPSAxNSwgMjYlKSwgMV5zdF4tb3JkZXIgcG9seW5vbWlhbCAoKm4qID0gMTEsIDE5JSksIGFuZCBmaW5hbGx5IDBedGheLW9yZGVyIHBvbHlub21pYWwgKCpuKiA9IDEwLCAxNyUpLg0KDQpGb3IgdGhlIHR3byBtb3N0IGNvbW1vbiBtb2RlbHMsIDJebmReLSBhbmQgM15yZF4tb3JkZXIgcG9seW5vbWlhbHMsIHRoaXMgY291bGQgc3VnZ2VzdCB0aGUgcHJlc2VuY2Ugb2YgYSBjcml0aWNhbCBveHlnZW4gdGhyZXNob2xkIChQY3JpdCkgd2hlcmUgdGhlIHJlbGF0aW9uc2hpcCBiZXR3ZWVuIE1PPHN1Yj4yPC9zdWI+IGFuZCBPPHN1Yj4yPC9zdWI+IGNoYW5nZXMuIFRvIGNvbmZpcm0gdGhlIHByZXNlbmNlIG9mIGEgUGNyaXQsIHdlIG5lZWQgdG8gdmFsaWRhdGVkIHRoZSBzaGFwZSBvZiB0aGUgcG9seW5vbWlhbHMgKHNlZSBQY3JpdCBtb2RlbCBiZWxvdykuIEluIGFueSBjYXNlLCB0aGVzZSB0eXBlIG9mIG1vZGVscyBhcmUgaW5kaWNhdGl2ZSBvZiAqKm94eXJlZ3VsYXRvcioqLg0KDQpUaGUgbmV4dCBtb3N0IGNvbW1vbiBtb2RlbCB3YXMgaXMgMV5zdF4tb3JkZXIgcG9seW5vbWlhbC4gSW4gdGhlIGNhc2Ugb2YgdGhlIDFec3ReLW9yZGVyIHBvbHlub21pYWxzLCBpdCBzdWdnZXN0IHRoZSBwcmVzZW5jZXMgb2YgbGluZWFyIHJlbGF0aW9uc2hpcCBiZXR3ZWVuIG8yIGFuZCBNTzIsIHdoaWNoIGlzIGluZGljYXRpdmUgb2YgKipveHljb25mb3JtZXIqKi4gSG93ZXZlciwgdG8gYmUgdHJ1ZSBldmlkZW5jZSBvZiBhIG94eWNvbmZvcm1lciB0aGlzIHJlbGF0aW9uc2hpcCBzaG91bGQgYmUgcG9zaXRpdmUgKGkuZS4gYXMgTzxzdWI+Mjwvc3ViPiBmYWxscyBNTzxzdWI+Mjwvc3ViPiBhbHNvIGZhbGxzKS4gMTMgb2YgdGhlIDE4IGluZGl2aWR1YWxzIGJlc3QgbW9kZWxsZWQgd2l0aCBhIGxpbmVhciBmdW5jdGlvbiBoYWQgcG9zaXRpdmUgZXN0aW1hdGVzIHdpdGggY3JlZGlibGUgaW50ZXJ2YWxzIHRoYXQgZGlkIG5vdCBvdmVybGFwIHdpdGggemVybyAoVGFibGUgUzMpLg0KDQpMYXN0bHksIDBedGheLW9yZGVyIHdhcyB0aGUgbGVhc3QgY29tbW9uICgqbiogPSAzLCA1JSksIGl0IHN1Z2dlc3RzIHRoYXQgTU88c3ViPjI8L3N1Yj4gZG9lcyBub3Qgc2hvdyBhIHN0YXRpc3RpY2FsbHkgc2lnbmlmaWNhbnQgZGVwZW5kZW5jZSBvbiB0aGUgTzxzdWI+Mjwvc3ViPi4gSW4gb3RoZXIgd29yZHMsIHRoZSBtZXRhYm9saWMgcmF0ZSBkb2VzIG5vdCBhZGp1c3QgYmFzZWQgb24gb3h5Z2VuIGF2YWlsYWJpbGl0eSwgYW5kIHRoZXJlIGlzIG5vIGNsZWFyIGNyaXRpY2FsIG94eWdlbiB0aHJlc2hvbGQgKFBjcml0KSB3aGVyZSB0aGUgcmVsYXRpb25zaGlwIGNoYW5nZXMuIFRoaXMgaXMgaW5kaWNhdGl2ZSBvZiBhICoqb3h5cmVndWxhdG9yKiouDQoNCmBgYHtyfQ0KYmVzdF9tb2QgPC0gYmVzdF9maXRfYmF5ZXNfcmVnICU+JSANCiAgZHBseXI6OmZpbHRlcihlbHBkX2xvb19yYW5rID09IDEpDQoNCnRvdGFsX2Zpc2ggPC0gbnJvdyhiZXN0X21vZCkNCg0KbW9kX3N1bW1hcnlfdGFibGUgPC0gYmVzdF9tb2QgJT4lIA0KICBkcGx5cjo6Z3JvdXBfYnkobW9kZWxfdHlwZSkgJT4lIA0KICBkcGx5cjo6cmVmcmFtZShuID0gbGVuZ3RoKGlkKSwNCiAgICAgICAgICAgICAgICAgcGVyY2VudCA9IHJvdW5kKChuL3RvdGFsX2Zpc2gpKjEwMCwyKSkgJT4lIA0KICBkcGx5cjo6bXV0YXRlKGJlc3RfbW9kZWxfbmFtZSA9IGNhc2Vfd2hlbigNCiAgICAgIG1vZGVsX3R5cGUgPT0gImxtXzAiIH4gIjB0aC1vcmRlciBwb2x5bm9taWFsIiwNCiAgICAgIG1vZGVsX3R5cGUgPT0gImxtXzEiIH4gIjFzdC1vcmRlciBwb2x5bm9taWFsIiwNCiAgICAgIG1vZGVsX3R5cGUgPT0gImxtXzIiIH4gIjJuZC1vcmRlciBwb2x5bm9taWFsIiwNCiAgICAgIG1vZGVsX3R5cGUgPT0gImxtXzMiIH4gIjNyZC1vcmRlciBwb2x5bm9taWFsIiwNCiAgICAgIFRSVUUgfiAiRVJST1IiDQogICAgKSkNCg0KDQp0YWJsZV9id20gPC0gbW9kX3N1bW1hcnlfdGFibGUgJT4lIA0KICBkcGx5cjo6c2VsZWN0KGJlc3RfbW9kZWxfbmFtZSwgZXZlcnl0aGluZygpLCAtbW9kZWxfdHlwZSkgJT4lIA0KICAgZ3QoKSAlPiUgDQogIGNvbHNfYWxpZ24oDQogICAgYWxpZ24gPSAiY2VudGVyIiwgDQogICAgY29sdW1ucyA9IGV2ZXJ5dGhpbmcoKQ0KICApDQoNCnRhYmxlX2J3bQ0KYGBgDQoNClN1bW1hcnkgb2YgZmlzaCBiZXN0IG1vZGVsIHdpdGggYSBsaW5lYXIgZnVuY3Rpb24uDQoNCiMjIyBUYWJsZSBTMw0KDQoqKlRhYmxlIFMzOioqIFRlbiBmaXNoIHRoYXQgd2VyZSBiZXN0IG1vZGVsbGVkIHdpdGggYSBsaW5lYXIgZnVuY3Rpb24sIHNob3dpbmcgcl4yXiwgYW5kIGVzdGltYXRlICgpLiBPbmx5IHR3byBmaXNoIGhhZCBwb3NpdGl2ZSBlc3RpbWF0ZXMgd2l0aCBjcmVkaWJsZSBpbnRlcnZhbHMgdGhhdCBkaWQgbm90IG92ZXJsYXAgd2l0aCB6ZXJvLCB3aGljaCBhcmUgaGlnaGxpZ2h0ZWQgYXMgY29uZm9ybWluZyBpbiB0aGUgdGFibGUuIFRodXMgaW4gdG90YWwsIDEzIG9mIDU4IGZpc2ggc2hvd2VkIHN1ZmZpY2llbnQgZXZpZGVuY2UgdG8gYmUgY29uZm9ybWluZy4NCg0KYGBge3J9DQp0YWJsZV9sbV8xIDwtIGJlc3RfbW9kICU+JSANCiAgZHBseXI6OmZpbHRlcihtb2RlbF90eXBlID09ICJsbV8xIikgJT4lIA0KICBkcGx5cjo6bXV0YXRlKHJfc3FfY2kgPSBwYXN0ZTAoZm9ybWF0KHJvdW5kKHIyLCAzKSwgc2NpZW50aWZpYyA9IEZBTFNFKSwgIiAoIiwgDQogICAgICAgICAgICAgICAgICAgZm9ybWF0KHJvdW5kKHIyX3EyLjUsIDMpLCBzY2llbnRpZmljID0gRkFMU0UpLCAiIHRvICIsIA0KICAgICAgICAgICAgICAgICAgIGZvcm1hdChyb3VuZChyMl9xOTcuNSwgMyksIHNjaWVudGlmaWMgPSBGQUxTRSksICIpIiksDQogIGVzdF9jaSA9IHBhc3RlMChmb3JtYXQocm91bmQoZXN0aW1hdGVfRE8sIDQpLCBzY2llbnRpZmljID0gRkFMU0UpLCAiICgiLCANCiAgICAgICAgICAgICAgICAgIGZvcm1hdChyb3VuZChjb25mLmxvd19ETywgNCksIHNjaWVudGlmaWMgPSBGQUxTRSksICIgdG8gIiwgDQogICAgICAgICAgICAgICAgICBmb3JtYXQocm91bmQoY29uZi5oaWdoX0RPLCA0KSwgc2NpZW50aWZpYyA9IEZBTFNFKSwgIikiKSwNCiAgICAgICAgICAgICAgICBjb25mb3JtZXIgPSBpZl9lbHNlKGNvbmYubG93X0RPID4gMCwgIkNvbmZvcm1pbmciLCAiTm90IGNvbmZvcm1pbmciKSkgJT4lIA0KICBkcGx5cjo6c2VsZWN0KGlkLCBlc3RpbWF0ZV9ETywgcl9zcV9jaSwgZXN0X2NpLCBjb25mb3JtZXIpICU+JSANCiAgZHBseXI6OmFycmFuZ2UoY29uZm9ybWVyLCBkZXNjKGVzdF9jaSkpICU+JSANCiAgZHBseXI6OnVuZ3JvdXAoKQ0KDQpmaXJzdF9vcmRlcl9pZHMgPC0gdGFibGVfbG1fMSAlPiUgDQogIGRpc3RpbmN0KGlkKSAlPiUgDQogIHB1bGwoLikNCg0KbWVhbl9tbzIgPC0gc2xvcGVfdGlkeSAlPiUgDQogIGRwbHlyOjpmaWx0ZXIoaWQgJWluJSBmaXJzdF9vcmRlcl9pZHMpICU+JSANCiAgZHBseXI6Omdyb3VwX2J5KGlkKSAlPiUgDQogIGRwbHlyOjpyZWZyYW1lKG1lYW5fTU8yX2cgPSBtZWFuKE1PMl9nLCBuYS5ybSA9IFRSVUUpKQ0KDQp0YWJsZV9sbV8xIDwtIGxlZnRfam9pbih0YWJsZV9sbV8xLCBtZWFuX21vMiwgYnkgPSAiaWQiKSANCg0KdGFibGVfbG1fMSA8LSAgdGFibGVfbG1fMSAlPiUgDQogIGRwbHlyOjptdXRhdGUocGVyY2VudF9jaGFuZ2UgPSAoZXN0aW1hdGVfRE8vbWVhbl9NTzJfZykqMTAwKSAlPiUgDQogIGRwbHlyOjpzZWxlY3QoaWQsIHJfc3FfY2ksIGVzdF9jaSwgcGVyY2VudF9jaGFuZ2UsIGNvbmZvcm1lcikNCg0KDQp0YWJsZV9sbV8xICU+JSANCiAgZ3QoKSAlPiUgDQogIGNvbHNfYWxpZ24oDQogICAgYWxpZ24gPSAiY2VudGVyIiwgDQogICAgY29sdW1ucyA9IGV2ZXJ5dGhpbmcoKQ0KICApICU+JSANCiAgY29sc19sYWJlbCgNCiAgICBpZCA9ICJGaXNoIElEIiwNCiAgICByX3NxX2NpID0gInIyIChDSSkiLA0KICAgIGVzdF9jaSA9ICJFc3RpbWF0ZSAoQ0kpIiwNCiAgICBwZXJjZW50X2NoYW5nZSA9ICJQZXJjZW50YWdlIGNoYW5nZSIsDQogICAgY29uZm9ybWVyID0gIkV2aWRlbmNlIG9mIG94eWNvbmZvcm1pbmciDQogICkNCmBgYA0KDQojIyMjIFBsb3RpbmcgYWxsIG1vZGVscw0KDQpEYXRhIHNldCB3aXRoIGFsbCBzbG9wZXMgYW5kIHdoaWNoIG1vZGVsIHdhcyBiZXN0DQoNCmBgYHtyfQ0KYmVzdF9maXQgPC0gbGVmdF9qb2luKHNsb3BlX3RpZHksIGJlc3RfbW9kLCBieSA9ICJpZCIpDQpgYGANCg0KU2F2aW5nIGFsbCByZWdyZXNzaW9uLCBhbmQgaGlnaGxpZ2h0aW5nIHRoZSBtb2RlbCB0aGF0IGhhcyB0aGUgYmVzdCBmaXQsIGJhc2VkIG9uIEFJQyB2YWx1ZXMuIFRoaXMgaXMgY2FsbGVkIGBjb21iaW5lZF9yZWdfcGxvdHMucGRmYA0KDQpgYGB7ciwgZmlnLmhlaWdodCA9IDMsIGZpZy53aWR0aD0gNH0NCiMgIyBDcmVhdGUgYSBsaXN0IHRvIHN0b3JlIHRoZSBwbG90cw0KIyBwbG90cyA8LSBsaXN0KCkNCiMgbW9kZWxfcHJlZHNfbGlzdCA8LSBsaXN0KCkNCiMgaW5jcmVtZW50YWxfcmVnX2JheWVzX3dkIDwtIHBhc3RlMChvdXRwdXRfZmlnX3dkLCAiLi9pbmNyZW1lbnRhbF9yZWdyZXNzaW9ucyIpDQojIA0KIyBmb3IgKGlkX2kgaW4gaWRzKSB7DQojICAgDQojICAgIyBGaWx0ZXIgZGF0YSBmb3IgdGhlIGN1cnJlbnQgSUQNCiMgICBkZl9pIDwtIGJheWVzX3JlZ19tb2RzX3ByZWRpY3Rpb25zICU+JQ0KIyAgICAgZHBseXI6OmZpbHRlcihpZCA9PSBpZF9pKSAlPiUgDQojICAgICBkcGx5cjo6bXV0YXRlKGxpbmVfc2l6ZSA9IGlmX2Vsc2UoZWxwZF9sb29fcmFuayA9PSAxLCAyLCAxKSwNCiMgICAgICAgICAgICBhbHBoYV92YWx1ZSA9IGlmX2Vsc2UoZWxwZF9sb29fcmFuayA9PSAxLCAxLCAwLjQpKQ0KIyAgIA0KIyAgIHhfbWluIDwtIGRmX2kgJT4lDQojICAgICBkcGx5cjo6cmVmcmFtZShtaW4gPSBtaW4oRE8pLCBuYS5ybSA9IFRSVUUpICU+JSANCiMgICAgIGRwbHlyOjpwdWxsKG1pbikNCiMgICANCiMgICB5X21heCA8LSBkZl9pICU+JQ0KIyAgICAgZHBseXI6OnJlZnJhbWUobWF4ID0gbWF4KE1PMl9nKSwgbmEucm0gPSBUUlVFKSAlPiUgDQojICAgICBkcGx5cjo6cHVsbChtYXgpDQojICAgDQojICAgYmVzdF93ZWlnaHRlZF9tb2RlbF9pIDwtIGJlc3RfZml0X2JheWVzX3JlZyAlPiUgDQojICAgICBkcGx5cjo6ZmlsdGVyKGlkID09IGlkX2kgJiBlbHBkX2xvb19yYW5rID09IDEpDQojICAgDQojICAgcG9seV9pX25hbWUgPC0gYmVzdF93ZWlnaHRlZF9tb2RlbF9pICU+JQ0KIyAgICAgZHBseXI6Om11dGF0ZShuYW1lID0gY2FzZV93aGVuKA0KIyAgICAgICBtb2RlbF90eXBlID09ICJsbV8wIiB+ICIwdGgtb3JkZXIiLA0KIyAgICAgICBtb2RlbF90eXBlID09ICJsbV8xIiB+ICIxc3Qtb3JkZXIiLA0KIyAgICAgICBtb2RlbF90eXBlID09ICJsbV8yIiB+ICIybmQtb3JkZXIiLA0KIyAgICAgICBtb2RlbF90eXBlID09ICJsbV8zIiB+ICIzcmQtb3JkZXIiLA0KIyAgICAgICBUUlVFIH4gIkVSUk9SIg0KIyAgICAgKSkgJT4lIA0KIyAgICAgZHBseXI6OnB1bGwobmFtZSkNCiMgICANCiMgICByX2kgPC0gYmVzdF93ZWlnaHRlZF9tb2RlbF9pICU+JSANCiMgICAgIGRwbHlyOjptdXRhdGUocl9zcV9jaSA9IHBhc3RlMChyb3VuZChyMiwgMyksICIgKCIsIA0KIyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICByb3VuZChyMl9xMi41LCAzKSwgIuKAkyIsIA0KIyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICByb3VuZChyMl9xOTcuNSwgMyksICIpIikpICU+JSANCiMgICAgIGRwbHlyOjpwdWxsKHJfc3FfY2kpDQojIA0KIyAgICMgQ3JlYXRlIHRoZSBwbG90DQojICAgcGxvdF9pIDwtIGdncGxvdCgpICsNCiMgICAgIGdlb21fcmliYm9uKGRhdGEgPSBkZl9pLA0KIyAgICAgICAgICAgICAgICAgYWVzKHggPSBETywgeSA9IHByZWRpY3RlZCwgeW1pbiA9IHByZWRfbG93ZXIsIHltYXggPSBwcmVkX3VwcGVyLCBmaWxsID0gbW9kZWxfdHlwZSksDQojICAgICAgICAgICAgICAgICBhbHBoYSA9IDAuMSkgKw0KIyAgICAgZ2VvbV9saW5lKGRhdGEgPSBkZl9pLCANCiMgICAgICAgICAgICAgICBhZXMoeCA9IERPLCB5ID0gcHJlZGljdGVkLCBjb2xvdXIgPSBtb2RlbF90eXBlLCBsaW5ld2lkdGggPSBsaW5lX3NpemUsIGFscGhhID0gYWxwaGFfdmFsdWUpKSArDQojICAgICBnZW9tX3BvaW50KGRhdGEgPSBkZl9pICU+JSBkcGx5cjo6ZmlsdGVyKGVscGRfbG9vX3JhbmsgPT0gMSksIGFlcyh4ID0gRE8sIHkgPSBNTzJfZyksIGFscGhhID0gMC42LCBjb2xvdXIgPSAiYmxhY2siLCBzaXplID0gMikgKw0KIyAgICAgc2NhbGVfY29sb3VyX21hbnVhbCh2YWx1ZXMgPSBjKCJyZWQiLCAiYmx1ZSIsICJncmVlbiIsICJwdXJwbGUiKSwgDQojICAgICAgICAgICAgICAgICAgICAgICAgIGxhYmVscyA9IGMoIjB0aCBPcmRlciIsICIxc3QgT3JkZXIiLCAiMm5kIE9yZGVyIiwgIjNyZCBPcmRlciIpKSArDQojICAgICBzY2FsZV9zaXplX2lkZW50aXR5KCkgKyAgIyBVc2UgdGhlIHNpemUgdmFsdWVzIGRpcmVjdGx5DQojICAgICBzY2FsZV9hbHBoYV9pZGVudGl0eShndWlkZSA9ICJub25lIikgKyAgIyBSZW1vdmUgdGhlIGFscGhhIGxlZ2VuZCANCiMgICAgIGFubm90YXRlKCJ0ZXh0IiwgeCA9IHhfbWluLCANCiMgICAgICAgICAgICAgIHkgPSB5X21heCwgDQojICAgICAgICAgICAgICBsYWJlbCA9IHBhc3RlMCgiQmVzdCBmaXQ6ICIscG9seV9pX25hbWUsICJcbiIsICJyMiA9ICIsIHJfaSksIA0KIyAgICAgICAgICAgICAgaGp1c3QgPSAwLCB2anVzdCA9IDEsIHNpemUgPSA0KSArDQojICAgICBsYWJzKA0KIyAgICAgICB0aXRsZSA9IHBhc3RlKGlkX2kpLA0KIyAgICAgICB4ID0gIkRpc3NvbHZlZCBveHlnZW4gcGVyY2VudGFnZSAoRE8pIiwNCiMgICAgICAgeSA9ICJNTzIgKG8yIG1nL2cvaCkiLA0KIyAgICAgICBjb2xvdXIgPSAiTW9kZWwiKSArDQojICAgICB0aGVtZV9jbGFzc2ljKCkgKw0KIyAgICAgdGhlbWUobGVnZW5kLnBvc2l0aW9uID0gIm5vbmUiKQ0KIyAgIA0KIyAgICMgU3RvcmUgdGhlIHBsb3QNCiMgICBwbG90c1tbaWRfaV1dIDwtIHBsb3RfaQ0KIyAgIA0KIyAgIHByaW50KHBsb3RfaSkNCiMgfQ0KIyANCiMgDQojIHBkZihmaWxlID0gcGFzdGUwKGluY3JlbWVudGFsX3JlZ19iYXllc193ZCwgIi4vY29tYmluZWRfcmVnX3Bsb3RzLnBkZiIpLCB3aWR0aCA9IDgsIGhlaWdodCA9IDYpDQojIA0KIyBkZXYub2ZmKCkgICMgQ2xvc2UgdGhlIFBERiBkZXZpY2UNCmBgYA0KDQpHZXR0aW5nIGEgcGxvdCBmb3IgZWFjaCBiZXN0IGZpdCByZWdyZXNzaW9uLCBhbmQgb3ZlcmxheWluZyBhIGdsb2JhbCBtb2RlbCBmb3IgdGhhdCBtb2RlbCB0eXBlLg0KDQpgYGB7ciwgZWNobz1GQUxTRX0NCm91dHB1dF9tb2RzX2JheWVzX2dsb2JhbF93ZCA8LSBwYXN0ZTAob3V0cHV0X21vZHNfd2QsICIuL2JheWVzLXJlZ3MtZ2xvYmFsIikNCmBgYA0KDQpIZXJlIHdlIGFyZSBncm91cGluZyBmaXNoIGJ5IGJlc3QgZml0dGluZyBtb2RlbCBhbmQgZ2V0dGluZyBhbiBhdmVyYWdlIHRyZW5kLiBJIGhhdmUgaGFzaGVkIHRoZSBjb2RlIHNvIHlvdSBkb24ndCBuZWVkIHRvIHJlLXJ1biB0aGUgbW9kZWxzDQoNCuKPre+4jyAqKlNraXAgdGhpcyBzdGVwKiogaWYgeW91IGhhdmUgZG93bmxvYWRlZCB0aGUgZ2xvYmFsIG1vZGVscyAqKicucmRzJyoqIGluIHRoZSAnYmF5ZXMtcmVncy1nbG9iYWwnIGZvbGRlcg0KDQpgYGB7cn0NCiMgc2VlZF9udW1iZXIgPSAxNDMwMTkNCiMgDQojICMgU2V0IHVwIHBhcmFsbGVsIGJhY2tlbmQNCiMgZnV0dXJlOjpwbGFuKG11bHRpc2Vzc2lvbiwgd29ya2VycyA9IDQpICAjIEFkanVzdCB3b3JrZXJzIGJhc2VkIG9uIHN5c3RlbSByZXNvdXJjZXMNCiMgDQojICMgRGVmaW5lIG1vZGVsIGZvcm11bGFzIGFuZCBkYXRhDQojIG1vZGVsX2Zvcm11bGFzIDwtIGxpc3QoDQojICAgbG1fMCA9IGJmKE1PMl9nIH4gMSwgZmFtaWx5ID0gZ2F1c3NpYW4oKSksDQojICAgbG1fMSA9IGJmKE1PMl9nIH4gRE8sIGZhbWlseSA9IGdhdXNzaWFuKCkpLA0KIyAgIGxtXzIgPSBiZihNTzJfZyB+IHBvbHkoRE8sIDIpLCBmYW1pbHkgPSBnYXVzc2lhbigpKSwNCiMgICBsbV8zID0gYmYoTU8yX2cgfiBwb2x5KERPLCAzKSwgZmFtaWx5ID0gZ2F1c3NpYW4oKSkNCiMgKQ0KIyANCiMgbW9kZWxfZGF0YSA8LSBsaXN0KA0KIyAgIGxtXzAgPSBiZXN0X2ZpdCAlPiUgZmlsdGVyKG1vZGVsX3R5cGUgPT0gImxtXzAiKSwNCiMgICBsbV8xID0gYmVzdF9maXQgJT4lIGZpbHRlcihtb2RlbF90eXBlID09ICJsbV8xIiksDQojICAgbG1fMiA9IGJlc3RfZml0ICU+JSBmaWx0ZXIobW9kZWxfdHlwZSA9PSAibG1fMiIpLA0KIyAgIGxtXzMgPSBiZXN0X2ZpdCAlPiUgZmlsdGVyKG1vZGVsX3R5cGUgPT0gImxtXzMiKQ0KIyApDQojIA0KIyAjIFJ1biBtb2RlbHMgaW4gcGFyYWxsZWwgd2l0aCBmdXR1cmVfbWFwKCkNCiMgbW9kZWxzIDwtIGZ1dHVyZV9tYXAobmFtZXMobW9kZWxfZm9ybXVsYXMpLCB+ew0KIyAgIGJybSgNCiMgICAgIGZvcm11bGEgPSBtb2RlbF9mb3JtdWxhc1tbLnhdXSwNCiMgICAgIGRhdGEgPSBtb2RlbF9kYXRhW1sueF1dLA0KIyAgICAgY29yZXMgPSA0LA0KIyAgICAgc2VlZCA9IHNlZWRfbnVtYmVyLA0KIyAgICAgc2F2ZV9wYXJzID0gc2F2ZV9wYXJzKGFsbCA9IFRSVUUpLA0KIyAgICAgc2FtcGxlX3ByaW9yID0gRkFMU0UsDQojICAgICBzaWxlbnQgPSBUUlVFLA0KIyAgICAgZmlsZSA9IHBhc3RlMChvdXRwdXRfbW9kc19iYXllc19nbG9iYWxfd2QsICIuLyIsIC54LCAiX2dsb2JhbCIpDQojICAgKQ0KIyB9KQ0KIyANCiMgIyBBc3NpZ24gbW9kZWwgbmFtZXMgdG8gcmVzdWx0cw0KIyBuYW1lcyhtb2RlbHMpIDwtIG5hbWVzKG1vZGVsX2Zvcm11bGFzKQ0KIyANCiMgIyBTdG9wIHBhcmFsbGVsIHBsYW4gYWZ0ZXIgZXhlY3V0aW9uDQojIGZ1dHVyZTo6cGxhbihzZXF1ZW50aWFsKQ0KYGBgDQoNCuKPre+4jyAqKlNraXAgdGhpcyBzdGVwKiogaWYgeW91IGhhdmUgZG93bmxvYWRlZCB0aGUgZ2xvYmFsX21vZGVsc19wcmVkX2RmLmNzdg0KDQpgYGB7cn0NCiMgZ2xvYmFsX21vZGVsc19wcmVkcyA8LSBsaXN0KCkNCiMgDQojIGZvciAobW9kZV9uYW1lIGluIG5hbWVzKGJheWVzX3JlZ19tb2RzX2dsb2JhbCkpIHsNCiMgICANCiMgICBtb2RfaSA8LSBiYXllc19yZWdfbW9kc19nbG9iYWxbW21vZGVfbmFtZV1dDQojICAgbW9kZV90eXBlID0gc3RyX3JlbW92ZShtb2RlX25hbWUsICJfZ2xvYmFsIikNCiMgICANCiMgICAgbW9kZWxfcHJlZGljdGlvbnNfaSA8LSBhcy5kYXRhLmZyYW1lKGZpdHRlZChtb2RfaSwgc3VtbWFyeSA9IFRSVUUpKSAlPiUgDQojICAgICAgIGRwbHlyOjptdXRhdGUobW9kZWwgPSBtb2RlX25hbWUsDQojICAgICAgICAgICAgICAgICAgICAgbW9kZWxfdHlwZSA9IG1vZGVfdHlwZSkgJT4lIA0KIyAgICAgICBkcGx5cjo6cmVuYW1lKHByZWRfbG93ZXIgPSBRMi41LCBwcmVkX3VwcGVyID0gUTk3LjUsIHByZWRpY3RlZCA9IEVzdGltYXRlLCBwcmVkX2Vycm9yID0gRXN0LkVycm9yKSAlPiUgDQojICAgICAgIGRwbHlyOjpzZWxlY3QobW9kZWwsIGV2ZXJ5dGhpbmcoKSkNCiMgICAgDQojICAgIG9yaWdpbmFsX2RhdGFfaSA8LSBiZXN0X2ZpdCAlPiUgZmlsdGVyKG1vZGVsX3R5cGUgPT0gbW9kZV90eXBlKSAlPiUgDQojICAgICAgZHBseXI6OnNlbGVjdCgtbW9kZWxfdHlwZSkNCiMgICAgIA0KIyAgICAgbW9kZWxfcHJlZGljdGlvbnNfb3JpZ2luYWxfaSA8LSBjYmluZChtb2RlbF9wcmVkaWN0aW9uc19pLCBvcmlnaW5hbF9kYXRhX2kpDQojICAgICANCiMgICAgDQojICAgIGdsb2JhbF9tb2RlbHNfcHJlZHNbW21vZGVfbmFtZV1dIDwtIG1vZGVsX3ByZWRpY3Rpb25zX29yaWdpbmFsX2kNCiMgfQ0KIyANCiMgZ2xvYmFsX21vZGVsc19wcmVkX2RmIDwtIGJpbmRfcm93cyhnbG9iYWxfbW9kZWxzX3ByZWRzKQ0KIyANCiMgd3JpdGUuY3N2KGdsb2JhbF9tb2RlbHNfcHJlZF9kZiwgIHBhc3RlMChtb2RfZGF0YV93ZCwgIi8uZ2xvYmFsX21vZGVsc19wcmVkX2RmLmNzdiIpLCByb3cubmFtZXMgPSBGQUxTRSkNCmBgYA0KDQpMb2FkIGRhdGEgZnJhbWUNCg0KYGBge3J9DQpnbG9iYWxfbW9kZWxzX3ByZWRfZGYgPC0gcmVhZC5jc3YocGFzdGUwKG1vZF9kYXRhX3dkLCAiLy5nbG9iYWxfbW9kZWxzX3ByZWRfZGYuY3N2IikpDQpgYGANCg0KIyMjIyBGaWd1cmUgUzkNCg0KQmVzdCBmaXQgcmVncmVzc2lvbnMgd2l0aCBnbG9iYWwgbW9kZWxzIGJhc2VkIG9uIHRoYXQgcG9seW5vbWlhbCBvcmRlci4NCg0KYGBge3J9DQpnbG9iYWxfbW9kZWxzX3ByZWRfZGYgPC0gZ2xvYmFsX21vZGVsc19wcmVkX2RmICU+JQ0KICBkcGx5cjo6bXV0YXRlKG1vZGVsX25hbWUgPSBjYXNlX3doZW4oDQogICAgICBtb2RlbF90eXBlID09ICJsbV8wIiB+ICIwdGgtb3JkZXIgcG9seW5vbWlhbCIsDQogICAgICBtb2RlbF90eXBlID09ICJsbV8xIiB+ICIxc3Qtb3JkZXIgcG9seW5vbWlhbCIsDQogICAgICBtb2RlbF90eXBlID09ICJsbV8yIiB+ICIybmQtb3JkZXIgcG9seW5vbWlhbCIsDQogICAgICBtb2RlbF90eXBlID09ICJsbV8zIiB+ICIzcmQtb3JkZXIgcG9seW5vbWlhbCIsDQogICAgICBUUlVFIH4gIkVSUk9SIg0KICAgICkpDQoNCmFubm90YXRpb25fZGF0YSA8LSBtb2Rfc3VtbWFyeV90YWJsZSAlPiUNCiAgZHBseXI6OnNlbGVjdChtb2RlbF90eXBlLCBiZXN0X21vZGVsX25hbWUsIG4pDQoNCmJheWVzX3JlZ19tb2RzX3ByZWRpY3Rpb25zX2Jlc3QgPC0gYmF5ZXNfcmVnX21vZHNfcHJlZGljdGlvbnMgJT4lIA0KICBkcGx5cjo6ZmlsdGVyKGVscGRfbG9vX3JhbmsgPT0gMSkgDQoNCmZpZ18yIDwtIGdncGxvdCgpICsNCiAgZ2VvbV9saW5lKGRhdGEgPSBiYXllc19yZWdfbW9kc19wcmVkaWN0aW9uc19iZXN0LA0KICAgICAgICAgICAgYWVzKHggPSBETywgeSA9IHByZWRpY3RlZCwgY29sb3IgPSBpZCksIHNpemUgPSAxKSArDQogICMgZ2VvbV9zbW9vdGgoZGF0YSA9IGJlc3RfZml0LA0KICAjICAgICAgICAgICBhZXMoeCA9IERPLCB5ID0gTU8yX2csIGNvbG9yID0gaWQpLCBzaXplID0gMSwgYWxwaGEgPSAxLCBzZSA9IEZBTFNFKSArDQogIGdlb21fcmliYm9uKGRhdGEgPSBnbG9iYWxfbW9kZWxzX3ByZWRfZGYsDQogICAgICAgICAgICAgIGFlcyh4ID0gRE8sIHltaW4gPSBwcmVkX2xvd2VyDQosIHltYXggPSBwcmVkX3VwcGVyLCBncm91cCA9IG1vZGVsKSwNCiAgICAgICAgICAgICAgZmlsbCA9ICIjRkM2Qzg1IiwgYWxwaGEgPSAwLjIpICsgICMgU2hhZGVkIGNvbmZpZGVuY2UgaW50ZXJ2YWxzDQogIGdlb21fbGluZShkYXRhID0gZ2xvYmFsX21vZGVsc19wcmVkX2RmLA0KICAgICAgICAgICAgYWVzKHggPSBETywgeSA9IHByZWRpY3RlZCksIGxpbmV3aWR0aCA9IDEuNSwgY29sb3IgPSAiI0ZGMDA3RiIpICsNCiAgZmFjZXRfd3JhcCh+bW9kZWxfdHlwZSkgKw0KICBzY2FsZV9jb2xvcl9ncmV5KHN0YXJ0ID0gMC4xLCBlbmQgPSAwLjkpICsNCiAgbGFicygNCiAgICAgIHRpdGxlID0gcGFzdGUoIkJlc3QgZml0IHJlZ3Jlc3Npb25zIGdyb3VwZWQgYnkgcG9seW5vbWlhbCBvcmRlciIpLA0KICAgICAgeCA9ICJEaXNzb2x2ZWQgb3h5Z2VuIHBlcmNlbnRhZ2UgKERPKSIsDQogICAgICB5ID0gIk1PMiAoTzIgbWcvZy9oKSIpICsNCiAgdGhlbWVfY2xhc3NpYygpICsNCiAgdGhlbWUobGVnZW5kLnBvc2l0aW9uID0gIm5vbmUiKSArDQogIGdlb21fdGV4dChkYXRhID0gYW5ub3RhdGlvbl9kYXRhLA0KICAgICAgICAgICAgYWVzKHggPSAtSW5mLCB5ID0gSW5mLCBsYWJlbCA9IHBhc3RlMCgiaXRhbGljKG4pID09ICIsIG4pKSwNCiAgICAgICAgICAgIGhqdXN0ID0gLTAuMSwgdmp1c3QgPSAxLjIsIGluaGVyaXQuYWVzID0gRkFMU0UsIHBhcnNlID0gVFJVRSkNCmZpZ18yDQpgYGANCg0KKipGaWd1cmUgUzk6KiogSW5kaXZpZHVhbCByZWdyZXNzaW9uIGN1cnZlcyBzaG93aW5nIHRoZSByZWxhdGlvbnNoaXAgYmV0d2VlbiBtYXNzLXNwZWNpZmljIG94eWdlbiBjb25zdW1wdGlvbiByYXRlICjhuYBP4oKCOyBtZ+KBu8K5IE/igoIgaOKBu8K5KSBhbmQgYW1iaWVudCBkaXNzb2x2ZWQgb3h5Z2VuIChETzsgJSBzYXR1cmF0aW9uKSwgZ3JvdXBlZCBieSB0aGUgYmVzdC1maXR0aW5nIHBvbHlub21pYWwgb3JkZXIgKDDhtZfKsCB0byAzyrPhtYgpLiBFYWNoIHBhbmVsIHJlcHJlc2VudHMgZmlzaCB3aG9zZSDhuYBP4oKC4oCTRE8gcmVsYXRpb25zaGlwIHdhcyBiZXN0IG1vZGVsbGVkIGJ5IHRoYXQgcG9seW5vbWlhbCBvcmRlciwgYmFzZWQgb24gbGVhdmUtb25lLW91dCBjcm9zcy12YWxpZGF0aW9uIChMT08tQ1YpLiBHcmV5IGxpbmVzIHJlcHJlc2VudCBpbmRpdmlkdWFsIG1vZGVsIGZpdHM7IHRoZSBib2xkIHBpbmsgbGluZSBzaG93cyB0aGUgZ3JvdXAtbGV2ZWwgdHJlbmQgd2l0aGluIGVhY2ggcG9seW5vbWlhbCBjbGFzcy4gRmlzaCB3ZXJlIG1vc3QgY29tbW9ubHkgYmVzdCBmaXQgYnkgYSAy4oG/4bWILW9yZGVyIHBvbHlub21pYWwgKG4gPSAyMiksIGZvbGxvd2VkIGJ5IDPKs+G1iC1vcmRlciAoKm4qID0gMTUpLCAxy6LhtZctb3JkZXIgKCpuKiA9IDExKSwgYW5kIDDhtZfKsC1vcmRlciAoKipuID0gMTApLiBIaWdoZXItb3JkZXIgZml0cyAoMuKBv+G1iCBhbmQgM8qz4bWIKSBhbmQgMOG1l8qwLW9yZGVyIGZpdHMgYXJlIGdlbmVyYWxseSBpbmRpY2F0aXZlIG9mIG94eXJlZ3VsYXRpbmcgcmVzcG9uc2VzLCB3aGlsZSAxy6LhtZctb3JkZXIgZml0cyBtYXkgc3VnZ2VzdCBveHljb25mb3JtaW5nIGJlaGF2aW91ciB3aGVuIHBvc2l0aXZlbHkgc2xvcGVkLiAgDQoNCg0KYGBge3J9DQojIGdnc2F2ZShmaWxlbmFtZSA9IHBhc3RlMChvdXRwdXRfZmlnX3dkLCAiLi9tc19maWd1cmVzLi9maWd1cmUtMi5wZGYiKSwgDQojICAgICAgICBwbG90ID0gZmlnXzIsDQojICAgICAgICB3aWR0aCA9IDIxMCwNCiMgICAgICAgIGhlaWdodCA9IDI5Ny8yLCAgIyBTcGVjaWZ5IHRoZSBoZWlnaHQgb2YgdGhlIHBsb3QNCiMgICAgICAgIHVuaXRzID0gIm1tIikNCmBgYA0KDQojIyBP4oKCY3JpdCBtb2RlbA0KDQpXZSB3aWxsIGNhbGN1bGF0ZSBP4oKCY3JpdCB1c2luZyBDaGFib3QgbWV0aG9kIGFuZCBmdW5jdGlvbiBgY2FsY08yY3JpdGAuICANCg0KVGhpcyBmdW5jdGlvbiB1c2VzIHRoZSBmaWZ0aCBwZXJjZW50aWxlIG9mIHRoZSBNTzxzdWI+Mjwvc3ViPiB2YWx1ZXMgb2JzZXJ2ZWQgYXQgZGlzc29sdmVkIG94eWdlbiBsZXZlbHMg4omlIDgwJSBhaXIgc2F0dXJhdGlvbiBhcyB0aGUgY3JpdGVyaW9uIHRvIGFzc2VzcyBsb3cgTU88c3ViPjI8L3N1Yj4gdmFsdWVzLiBUaGUgYWxnb3JpdGhtIHRoZW4gaWRlbnRpZmllcyBhbGwgdGhlIE1PPHN1Yj4yPC9zdWI+IG1lYXN1cmVtZW50cyBncmVhdGVyIHRoYW4gdGhpcyBtaW5pbWFsbHkgYWNjZXB0YWJsZSBNTzxzdWI+Mjwvc3ViPiB2YWx1ZS4gV2l0aGluIHRoaXMgc3ViLXNldCwgaXQgaWRlbnRpZmllcyB0aGUgTU88c3ViPjI8L3N1Yj4yIG1lYXN1cmVtZW50IG1hZGUgYXQgdGhlIGxvd2VzdCBETyBhbmQgdGhlcmVhZnRlciBjb25zaWRlcnMgdGhpcyBETyBhcyBjYW5kaWRhdGUgZm9yIGJyZWFrcG9pbnQgKG5hbWVkIHBpdm90RE8gaW4gdGhlIHNjcmlwdCkuIEEgcmVncmVzc2lvbiBpcyB0aGVuIGNhbGN1bGF0ZWQgdXNpbmcgb2JzZXJ2YXRpb25zIGF0IERPIGxldmVscyBcPCBwaXZvdERPLCBhbmQgYSBmaXJzdCBlc3RpbWF0ZSBvZiBPPHN1Yj4yY3JpdDwvc3ViPiBpcyBjYWxjdWxhdGVkIGFzIHRoZSBpbnRlcnNlY3Rpb24gb2YgdGhpcyByZWdyZXNzaW9uIGxpbmUgd2l0aCB0aGUgaG9yaXpvbnRhbCBsaW5lIHJlcHJlc2VudGluZyBTTVIuIFRoZSBzY3JpcHQgdGhlbiBnb2VzIHRocm91Z2ggdmFsaWRhdGlvbiBzdGVwcyB0byBlbnN1cmUgdGhhdCB0aGUgc2xvcGUgb2YgdGhlIHJlZ3Jlc3Npb24gaXMgbm90IHNvIGxvdyB0aGF0IHRoZSBsaW5lLCBwcm9qZWN0ZWQgdG8gbm9ybW94aWMgRE8gbGV2ZWxzLCBwYXNzZXMgYmVsb3cgYW55IE1PPHN1Yj4yPC9zdWI+IHZhbHVlcyBvYnNlcnZlZCBpbiBub3Jtb3hpYS4gSXQgYWxzbyBlbnN1cmVzIHRoYXQgdGhlIGludGVyY2VwdCBpcyBub3QgZ3JlYXRlciB0aGFuIHplcm8uIENvcnJlY3RpdmUgbWVhc3VyZXMgYXJlIHRha2VuIGlmIHN1Y2ggcHJvYmxlbXMgYXJlIGVuY291bnRlcmVkLiAgDQoNCmxvd2VzdE1PMiBkZWZhdWx0IGlzIHRoZSBgcXVhbnRpbGUoTU8yW0RPID49IDgwXSwgcD0wLjA1KWAuIEl0IGlzIHVzZWQgdG8gc2VnbWVudCB0aGUgZGF0YSBhbmQgbG9jYXRlIHRoZSBwaXZvdERPLiAgDQoNCmBgYHtyfQ0KaWRzIDwtIHNsb3BlX3RpZHkgJT4lIA0KICBkcGx5cjo6ZGlzdGluY3QoaWQpICU+JSANCiAgZHBseXI6OnB1bGwoKQ0KDQpwY3JpdF9tb2RlbF9kZl9saXN0IDwtIGxpc3QoKQ0KcGNyaXRfbW9kZWxzIDwtICBsaXN0KCkNCg0KZm9yIChpZF9pIGluIGlkcykgew0KDQpkZl9pIDwtIHNsb3BlX3RpZHkgJT4lIA0KICBkcGx5cjo6ZmlsdGVyKGlkID09IGlkX2kpDQoNCm8yY3JpdCA8LSBjYWxjTzJjcml0KERhdGEgPSBkZl9pLCBTTVIgPSBkZl9pJFNNUlsxXSwgbG93ZXN0TU8yPU5BLCBnYXBMaW1pdCA9IDQsDQogICAgICAgICAgICAgICAgICAgICBtYXgubmIuTU8yLmZvci5yZWcgPSA3KQ0KDQpiZXN0X21vZF90eXBlIDwtIGJlc3RfbW9kICU+JQ0KICBkcGx5cjo6ZmlsdGVyKGlkID09IGlkX2kpICU+JSANCiAgcHVsbChtb2RlbF90eXBlKQ0KICANCnZhdWxlIDwtIG8yY3JpdCRvMmNyaXQNCmxvd2VzdE1PMiA9IHF1YW50aWxlKGRmX2kkTU8yW2RmX2kkRE8gPj0gODBdLCBwPTAuMDUpDQpTTVIgPC0gbzJjcml0JFNNUg0KbmJfbW8yX2NvbmZvcm1pbmcgPC0gbzJjcml0JE5iX01PMl9jb25mb3JtaW5nDQpyMiA8LSBvMmNyaXQkcjINCm1ldGhvZCA8LSBvMmNyaXQkTWV0aG9kDQpwIDwtIG8yY3JpdCRQWzFdDQpsZXRoYWxfcG9pbnQgPC0gbWluKG8yY3JpdCRsZXRoYWxQb2ludHMpDQoNCnBjcml0X21vZGVsX2RmIDwtIHRpYmJsZSgNCiAgICAgIGlkID0gaWRfaSwNCiAgICAgIHBjcml0X3ZhdWxlID0gdmF1bGUsDQogICAgICBwY3JpdF9zbXIgPSBTTVIsDQogICAgICBwY3JpdF9sb3dlc3RNTzIgPSBsb3dlc3RNTzIsDQogICAgICBwY3JpdF9uYl9tbzJfY29uZm9ybWluZyA9IG5iX21vMl9jb25mb3JtaW5nLA0KICAgICAgcGNyaXRfbGV0aGFsX3BvaW50cyA9IGxldGhhbF9wb2ludCwNCiAgICAgIHBjcml0X3IyID0gcjIsDQogICAgICBwY3JpdF9tZXRob2QgPSBtZXRob2QsDQogICAgICBwY3JpdF9wID0gcCwNCiAgICAgIGJlc3RfbW9kX3R5cGUgPSBiZXN0X21vZF90eXBlDQogICAgKQ0KDQpwY3JpdF9tb2RlbF9kZl9saXN0W1tpZF9pXV0gPC0gcGNyaXRfbW9kZWxfZGYNCg0KcGNyaXRfbW9kZWxzW1tpZF9pXV0gPC0gbzJjcml0DQoNCn0NCg0KcGNyaXRfbW9kZWxfZGYgPC0gYmluZF9yb3dzKHBjcml0X21vZGVsX2RmX2xpc3QpDQpgYGANCg0KIyMjIFBsb3RpbmcgTzJjcml0DQoNCkhlcmUncyB3ZSBhcmUgc2F2aW5nIGFsbCBPPHN1Yj4yY3JpdDwvc3ViPiBtb2RlbHMsIGV2ZW4gdGhvc2UgTU88c3ViPjI8L3N1Yj4tTzxzdWI+Mjwvc3ViPiByZWxhdGlvbnNoaXBzIGJlc3QgbW9kZWxsZWQgYnkgMF50aF4tIGFuZCAxXnN0Xi1vcmRlciBwb2x5bm9taWFscy4NCg0KYGBge3J9DQojICMgQ3JlYXRlIG91dHB1dCBkaXJlY3RvcnkgaWYgbmVlZGVkDQojIG91dHB1dF9maWdfcGNyaXRfY2hhYm90X3dkIDwtIGZpbGUucGF0aChvdXRwdXRfZmlnX3dkLCAibW9kZWxfY2hhYm90IikNCiMgaWYgKCFkaXIuZXhpc3RzKG91dHB1dF9maWdfcGNyaXRfY2hhYm90X3dkKSkgew0KIyAgIGRpci5jcmVhdGUob3V0cHV0X2ZpZ19wY3JpdF9jaGFib3Rfd2QpDQojIH0NCiMgDQojIGlkcyA8LSBzbG9wZV90aWR5ICU+JSANCiMgICBkcGx5cjo6ZGlzdGluY3QoaWQpICU+JSANCiMgICBkcGx5cjo6cHVsbCgpDQojIA0KIyBwY3JpdF9jaGFib3RfbGlzdCA8LSBsaXN0KCkNCiMgDQojICMgT3BlbiBhIHNpbmdsZSBQREYgZGV2aWNlDQojIHBkZihmaWxlID0gZmlsZS5wYXRoKG91dHB1dF9maWdfcGNyaXRfY2hhYm90X3dkLCAiY29tYmluZWRfY2hhYm90X3Bsb3RzLnBkZiIpLCANCiMgICAgIHdpZHRoID0gOCwgaGVpZ2h0ID0gNikNCiMgDQojIGZvciAoaWRfaSBpbiBpZHMpIHsNCiMgICANCiMgICBiZXN0X21vZF90eXBlIDwtIGJlc3RfbW9kICU+JQ0KIyAgIGRwbHlyOjpmaWx0ZXIoaWQgPT0gaWRfaSkgJT4lIA0KIyAgIHB1bGwobW9kZWxfdHlwZSkNCiMgICANCiMgICByMiA8LSBwY3JpdF9tb2RlbF9kZiAlPiUgDQojICAgICBkcGx5cjo6ZmlsdGVyKGlkID09IGlkX2kpICU+JSANCiMgICAgIGRwbHlyOjptdXRhdGUocGNyaXRfcjIgPSByb3VuZChwY3JpdF9yMiwgMykpICU+JSANCiMgICAgIGRwbHlyOjpwdWxsKHBjcml0X3IyKQ0KIyAgIA0KIyAgIGNvbmZvcm1pbmcgPC0gcGNyaXRfbW9kZWxfZGYgJT4lIA0KIyAgICAgZHBseXI6OmZpbHRlcihpZCA9PSBpZF9pKSAlPiUgDQojICAgICBkcGx5cjo6bXV0YXRlKHBjcml0X25iX21vMl9jb25mb3JtaW5nID0gcm91bmQocGNyaXRfbmJfbW8yX2NvbmZvcm1pbmcsIDMpKSAlPiUgDQojICAgICBkcGx5cjo6cHVsbChwY3JpdF9uYl9tbzJfY29uZm9ybWluZykNCiMgICANCiMgICBQIDwtIHBjcml0X21vZGVsX2RmICU+JSANCiMgICAgIGRwbHlyOjpmaWx0ZXIoaWQgPT0gaWRfaSkgJT4lIA0KIyAgICAgZHBseXI6Om11dGF0ZShwY3JpdF9wID0gcm91bmQocGNyaXRfcCwgMykpICU+JSANCiMgICAgIGRwbHlyOjpwdWxsKHBjcml0X3ApDQojICAgDQojICAgU01SIDwtIHBjcml0X21vZGVsX2RmICU+JSANCiMgICAgIGRwbHlyOjpmaWx0ZXIoaWQgPT0gaWRfaSkgJT4lIA0KIyAgICAgZHBseXI6Om11dGF0ZShwY3JpdF9zbXIgPSByb3VuZChwY3JpdF9zbXIsIDMpKSAlPiUgDQojICAgICBkcGx5cjo6cHVsbChwY3JpdF9zbXIpDQojICAgDQojICAgbG93ZXN0TU8yIDwtIHBjcml0X21vZGVsX2RmICU+JSANCiMgICAgIGRwbHlyOjpmaWx0ZXIoaWQgPT0gaWRfaSkgJT4lIA0KIyAgICAgZHBseXI6Om11dGF0ZShwY3JpdF9sb3dlc3RNTzIgPSByb3VuZChwY3JpdF9sb3dlc3RNTzIsIDMpKSAlPiUgDQojICAgICBkcGx5cjo6cHVsbChwY3JpdF9sb3dlc3RNTzIpDQojICAgDQojICAgIyBHZW5lcmF0ZSBhbmQgcmVuZGVyIHRoZSBwbG90DQojICAgcGxvdE8yY3JpdChvMmNyaXRvYmogPSBwY3JpdF9tb2RlbHNbW2lkX2ldXSkNCiMgICANCiMgICAjIEFkZCBhIHRpdGxlDQojICAgbXRleHQoDQojICAgICB0ZXh0ID0gcGFzdGUwKGlkX2kpLA0KIyAgICAgc2lkZSA9IDMsIGxpbmUgPSAyLCBhZGogPSAwLA0KIyAgICAgY29sID0gImJsdWUiLCBmb250ID0gMiwgY2V4ID0gMS4yDQojICAgKQ0KIyAgIA0KIyAgIG10ZXh0KA0KIyAgICAgdGV4dCA9IHBhc3RlMCgiUjIgPSAiLCByMiwgIjsgcCA9ICIsIFAsICI7IENQIDwgU01SID0gIiwgY29uZm9ybWluZywgIjsgU01SID0gIiwgU01SLCAiOyBsb3dlc3RNTzIgPSAiLGxvd2VzdE1PMiksDQojICAgICBzaWRlID0gMywgbGluZSA9IDEsIGFkaiA9IDAsDQojICAgICBjb2wgPSAiYmx1ZSIsIGZvbnQgPSAxLCBjZXggPSAwLjgNCiMgICApDQojICAgDQojICAgbXRleHQoDQojICAgICB0ZXh0ID0gcGFzdGUwKCJCZXN0IG1vZGVsIGZpdCA9ICIsIGJlc3RfbW9kX3R5cGUpLA0KIyAgICAgc2lkZSA9IDMsIGxpbmUgPSAwLCBhZGogPSAwLA0KIyAgICAgY29sID0gInJlZCIsIGZvbnQgPSAxLCBjZXggPSAwLjgNCiMgICApDQojIH0NCiMgDQojICMgQ2xvc2UgdGhlIFBERiBkZXZpY2UgKmFmdGVyKiB0aGUgbG9vcA0KIyBkZXYub2ZmKCkNCmBgYA0KDQpOb3cgcHJpbnRpbmcgdGhlIGZpc2ggdGhhdCB3ZXJlIGZpdHRlZCB3aXRoIGhpZ2hlciBvcmRlci1wb2x5bm9taWFscyBpbiB0aGUgaHRsbSBkb2N1bWVudA0KDQpgYGB7ciwgZmlnLmhlaWdodCA9IDMsIGZpZy53aWR0aD0gOH0NCg0KcGFyKG1mcm93ID0gYygxLCAyKSwgbWFyID0gYyg0LCA0LCA0LCAyKSkgDQoNCnBjcml0X21vZGVsX21ldGEgPC0gc2xvcGVfdGlkeSAlPiUgDQogIGRwbHlyOjpncm91cF9ieShpZCkgJT4lIA0KICBkcGx5cjo6c2xpY2UoMSkgJT4lIA0KICBkcGx5cjo6dW5ncm91cCgpICU+JSANCiAgZHBseXI6OmxlZnRfam9pbihwY3JpdF9tb2RlbF9kZiwgLiwgYnkgPSAiaWQiKQ0KDQpoaWdoZXJfb3JkZXJfaWRzIDwtICBwY3JpdF9tb2RlbF9tZXRhICU+JSANCiAgZHBseXI6OmZpbHRlcihiZXN0X21vZF90eXBlID09ICJsbV8yIiB8IGJlc3RfbW9kX3R5cGUgPT0gImxtXzMiKSAlPiUgDQogIGRwbHlyOjpkaXN0aW5jdChpZCkgJT4lIA0KICBkcGx5cjo6cHVsbCgpDQoNCmZvciAoaWRfaSBpbiBoaWdoZXJfb3JkZXJfaWRzKSB7DQogIA0KICBjb21tZW50IDwtIHBjcml0X21vZGVsX21ldGEgJT4lIA0KICAgIGRwbHlyOjpmaWx0ZXIoaWQgPT0gaWRfaSkgJT4lIA0KICAgIGRwbHlyOjpzbGljZSgxKSAlPiUgDQogICAgZHBseXI6Om11dGF0ZShjb21tZW50ID0gaWZfZWxzZShpcy5uYShjb21tZW50cyksICIiLCBwYXN0ZTAoIiMiLCBjb21tZW50cykpKSAlPiUgDQogICAgcHVsbChjb21tZW50KQ0KICANCiAgcjIgPC0gcGNyaXRfbW9kZWxfbWV0YSAlPiUgDQogICAgZHBseXI6OmZpbHRlcihpZCA9PSBpZF9pKSAlPiUgDQogICAgZHBseXI6Om11dGF0ZShwY3JpdF9yMiA9IHJvdW5kKHBjcml0X3IyLCAzKSkgJT4lIA0KICAgIGRwbHlyOjpwdWxsKHBjcml0X3IyKQ0KICANCiAgY29uZm9ybWluZyA8LSBwY3JpdF9tb2RlbF9tZXRhICU+JSANCiAgICBkcGx5cjo6ZmlsdGVyKGlkID09IGlkX2kpICU+JSANCiAgICBkcGx5cjo6bXV0YXRlKHBjcml0X25iX21vMl9jb25mb3JtaW5nID0gcm91bmQocGNyaXRfbmJfbW8yX2NvbmZvcm1pbmcsIDMpKSAlPiUgDQogICAgZHBseXI6OnB1bGwocGNyaXRfbmJfbW8yX2NvbmZvcm1pbmcpDQogIA0KICBQIDwtIHBjcml0X21vZGVsX21ldGEgJT4lIA0KICAgIGRwbHlyOjpmaWx0ZXIoaWQgPT0gaWRfaSkgJT4lIA0KICAgIGRwbHlyOjptdXRhdGUocGNyaXRfcCA9IHJvdW5kKHBjcml0X3AsIDMpKSAlPiUgDQogICAgZHBseXI6OnB1bGwocGNyaXRfcCkNCiAgDQogIFNNUiA8LSBwY3JpdF9tb2RlbF9tZXRhICU+JSANCiAgICBkcGx5cjo6ZmlsdGVyKGlkID09IGlkX2kpICU+JSANCiAgICBkcGx5cjo6bXV0YXRlKHBjcml0X3NtciA9IHJvdW5kKHBjcml0X3NtciwgMykpICU+JSANCiAgICBkcGx5cjo6cHVsbChwY3JpdF9zbXIpDQogIA0KICBsb3dlc3RNTzIgPC0gcGNyaXRfbW9kZWxfbWV0YSAlPiUgDQogICAgZHBseXI6OmZpbHRlcihpZCA9PSBpZF9pKSAlPiUgDQogICAgZHBseXI6Om11dGF0ZShwY3JpdF9sb3dlc3RNTzIgPSByb3VuZChwY3JpdF9sb3dlc3RNTzIsIDMpKSAlPiUgDQogICAgZHBseXI6OnB1bGwocGNyaXRfbG93ZXN0TU8yKQ0KICANCiAgIyBHZW5lcmF0ZSBhbmQgcmVuZGVyIHRoZSBwbG90DQogIHBsb3RPMmNyaXQobzJjcml0b2JqID0gcGNyaXRfbW9kZWxzW1tpZF9pXV0pDQogIA0KICAjIEFkZCBhIHRpdGxlDQogIG10ZXh0KA0KICAgIHRleHQgPSBwYXN0ZTAoaWRfaSwgIiAiLCBjb21tZW50KSwNCiAgICBzaWRlID0gMywgbGluZSA9IDIsIGFkaiA9IDAsDQogICAgY29sID0gImJsdWUiLCBmb250ID0gMiwgY2V4ID0gMS4yDQogICkNCiAgDQogIG10ZXh0KA0KICAgIHRleHQgPSBwYXN0ZTAoIlIyID0gIiwgcjIsICI7IHAgPSAiLCBQLCAiOyBDUCA8IFNNUiA9ICIsIGNvbmZvcm1pbmcpLA0KICAgIHNpZGUgPSAzLCBsaW5lID0gMSwgYWRqID0gMCwNCiAgICBjb2wgPSAiYmx1ZSIsIGZvbnQgPSAwLjUsIGNleCA9IDAuOA0KICApDQogIA0KICAgbXRleHQoDQogICAgdGV4dCA9IHBhc3RlMCgiU01SID0gIiwgU01SLCAiOyBsb3dlc3RNTzIgPSAiLGxvd2VzdE1PMiksDQogICAgc2lkZSA9IDMsIGxpbmUgPSAwLCBhZGogPSAwLA0KICAgIGNvbCA9ICJibHVlIiwgZm9udCA9IDAuNSwgY2V4ID0gMC44DQogICkNCn0NCmBgYA0KDQoNCiMjIyBWaXN1YWwgaW5zcGVjdGlvbg0KDQpXZSBjb25kdWN0ZWQgdmlzdWFsIGluc3BlY3Rpb24gb2YgYWxsIDU4IGZpc2ggdG8gZGV0ZXJtaW5lIGlmIE88c3ViPjJjcml0czwvc3ViPiB3ZXJlIHByZXNlbnQuIFdlIGhhdmUgbG9hZGVkIHRoaXMgYXMgYSBkYXRhIGZyYW1lICoq8J+SvyBvMmNyaXRfY2hlY2sqKg0KDQpGb3IgdGhpcyB2aXN1YWwgaW5zcGVjdGlvbiwgd2UgZ3JvdXBlZCBmaXNoIGludG8gIm5vIiwgIm1heWJlIiBvciAieWVzIiwgYmFzZWQgb24gdGhlIGNlcnRhaW50eSBhdCB3aGljaCB3ZSBvYnNlcnZlZCBhIE88c3ViPjJjcml0czwvc3ViPi4gSW4gdGhlIGNhc2Ugb2YgeWVzIG9yIG1heWJlLCB3ZSBhbHNvIGVzdGltYXRlIGRpc3NvbHZlZCBveHlnZW4gcGVyY2VudGFnZS4gQWxsIGF1dGhvcnMgZGlkIHRoaXMgaW5kZXBlbmRlbnRseS4gIA0KDQojIyMjIPCfk4ggUmVzdWx0cw0KDQpMZXQncyBmaWd1cmUgb3V0IGhvdyBjb25zaXN0ZW50IHRoZSBjYXRlZ29yaXNhdGlvbiB3YXMsIGFuZCB3aGF0IHRoZSBtYWpvcml0eSByZXNwb25zZSB3YXMuIFRoZXJlIHdlcmUgMzUgY2FzZXMgd2l0aCBubyBkaXNhZ3JlZW1lbnRzICg2MC4zNCUpLCAyMyBjYXNlcyB3aXRoIG9uZSBvciBtb3JlIGRpc2FncmVlbWVudHMgKDM5LjY2JSkuICANCg0KYGBge3J9DQpkaXNhZ3JlZW1lbnRfc3VtbWFyeSA8LSBvMmNyaXRfY2hlY2sgJT4lDQogIGdyb3VwX2J5KGlkKSAlPiUNCiAgZHBseXI6OnJlZnJhbWUoDQogICAgbl9kaXNhZ3JlZW1lbnQgPSBuX2Rpc3RpbmN0KHBjcml0X2NhdCktMSwNCiAgICBjYXRfZGlzYWdyZWVtZW50ID0gaWZfZWxzZShuX2Rpc2FncmVlbWVudD09MCwgMCwgMSksDQogICAgbWFqb3JpdHlfY2F0ID0gbmFtZXMoc29ydCh0YWJsZShwY3JpdF9jYXQpLCBkZWNyZWFzaW5nID0gVFJVRSkpWzFdLA0KICAgIGNlcnRhaW50eSA9IG1heCh0YWJsZShwY3JpdF9jYXQpLzUqMTAwKQ0KICApDQoNCnRvdGFsX2Fzc2VzbWVudCA8LSBvMmNyaXRfY2hlY2sgJT4lIA0KICBkaXN0aW5jdChpZCkgJT4lIA0KICBucm93KCkNCg0KZGlzYWdyZWVtZW50X3N1bW1hcnkgJT4lDQogIGdyb3VwX2J5KGNhdF9kaXNhZ3JlZW1lbnQpICU+JSANCiAgZHBseXI6OnJlZnJhbWUoDQogICAgbiA9IGxlbmd0aChpZCksDQogICAgcGVyY2VudCA9IHJvdW5kKG4vdG90YWxfYXNzZXNtZW50KjEwMCwyKQ0KICApICU+JSANCiAgZ3QoKQ0KYGBgDQoNCiMjIyMgRmlndXJlIFMxMA0KDQpIZXJlJ3MgYSBoZWF0IG1hcCB0byBzaG93IGRlY2lzaW9ucyBmb3IgZWFjaCBmaXNoDQoNCmBgYHtyLCBmaWcuaGVpZ2h0PTEwLCBmaWcud2lkdGg9NX0NCiMgRGVmaW5lIGN1c3RvbSBjb2xvdXJzIGZvciBwY3JpdF9jYXQNCmNhdGVnb3J5X2NvbG91cnMgPC0gYygNCiAgIm5vIiA9ICIjRTAzQzMyIiwNCiAgIm1heWJlIiA9ICIjRkZEMzAxIiwNCiAgInllcyIgPSAiIzdCQjY2MiINCikNCg0KIyBDcmVhdGUgaGVhdG1hcA0KZ2dwbG90KG8yY3JpdF9jaGVjaywgYWVzKHggPSBleHBlcnQsIHkgPSBpZCwgZmlsbCA9IHBjcml0X2NhdCkpICsNCiAgZ2VvbV90aWxlKGNvbG91ciA9ICJ3aGl0ZSIpICsNCiAgc2NhbGVfZmlsbF9tYW51YWwodmFsdWVzID0gY2F0ZWdvcnlfY29sb3VycywgbmEudmFsdWUgPSAiZ3JleTkwIikgKw0KICBsYWJzKA0KICAgIHRpdGxlID0gIlZpc3VhbCBBc3Nlc3NtZW50IG9mIHBjcml0X2NhdCBieSBFeHBlcnQgYW5kIElEIiwNCiAgICB4ID0gIkV4cGVydCIsDQogICAgeSA9ICJGaXNoIElEIiwNCiAgICBmaWxsID0gIkNhdGVnb3J5Ig0KICApICsNCiAgdGhlbWVfbWluaW1hbCgpICsNCiAgdGhlbWUoDQogICAgYXhpcy50ZXh0LnkgPSBlbGVtZW50X3RleHQoc2l6ZSA9IDcpLCAjIGFkanVzdCBzaXplIGFzIG5lZWRlZA0KICAgIGF4aXMudGV4dC54ID0gZWxlbWVudF90ZXh0KGFuZ2xlID0gNDUsIGhqdXN0ID0gMSksDQogICAgcGFuZWwuZ3JpZCA9IGVsZW1lbnRfYmxhbmsoKQ0KICApDQpgYGANCg0KRXhwZXJ0IGRldmlhdGlvbiBmcm9tIHRoZSBtYWpvcml0eSBhc3Nlc3NtZW50IHJhbmdlZCBmcm9tIDIyLjQxIHRvIDYuOTAlLg0KDQpgYGB7cn0NCm8yY3JpdF9jaGVja19zdW0gPC0gbzJjcml0X2NoZWNrICU+JSANCiAgZHBseXI6OmxlZnRfam9pbiguLCBkaXNhZ3JlZW1lbnRfc3VtbWFyeSwgYnkgPSAiaWQiKSAlPiUgDQogIGRwbHlyOjptdXRhdGUoZGV2aWF0ZWQgPSBpZl9lbHNlKHBjcml0X2NhdCAhPSBtYWpvcml0eV9jYXQsIDEsIDApKQ0KDQpvMmNyaXRfY2hlY2tfc3VtICU+JQ0KICBncm91cF9ieShleHBlcnQpICU+JQ0KICBzdW1tYXJpc2UoDQogICAgZGV2aWF0aW9ucyA9IHN1bShkZXZpYXRlZCksDQogICAgZGV2aWF0aW9uX3JhdGUgPSByb3VuZChkZXZpYXRpb25zIC8gNTggKiAxMDAsIDIpDQogICkgJT4lIA0KICBkcGx5cjo6YXJyYW5nZShkZXNjKGRldmlhdGlvbnMpKSAlPiUgDQogIGd0KCkNCmBgYA0KDQpMZXQncyBtYWtlIGEgbGlzdCBvZiB0aG9zZSB0aGF0IGhhZCBtYWpvcml0eSBhc3NpZ25lZCBhcyB5ZXMgZm9yIGFuIE88c3ViPjJjcml0PC9zdWI+LiBUaGVyZSB3YXMgMTAgdG90YWwuICANCg0KYGBge3J9DQpvMmNyaXRfdmlzdWFsX3llcyA8LSAgZGlzYWdyZWVtZW50X3N1bW1hcnkgJT4lIA0KICBkcGx5cjo6ZmlsdGVyKG1ham9yaXR5X2NhdCA9PSAieWVzIikgJT4lIA0KICBkcGx5cjo6cHVsbChpZCkNCg0KcGFzdGUwKCJCYXNlZCBvbiB0aGUgbWFqb3JpdHkgdmlzdWFsIGFzc2Vzc21lbnQgIiwgbGVuZ3RoKG8yY3JpdF92aXN1YWxfeWVzKSwgIiBmaXNoIGhhdmUgcG9zc2libGUgT+KCgmNyaXQuIikNCmBgYA0KDQojIyMgT+KCgmNyaXQgbnVtZXJpY2FsIHJ1bGVzDQoNCldlIGFsc28gdXNlZCBhIG51bWVyaWNhbCBydWxlIHRvIGFzc2VzIGlmIGEgTzxzdWI+MmNyaXQ8L3N1Yj4gd2FzIG9ic2VydmVkLg0KDQpXZSBmaWx0ZXJlZCBmb3Igb25seSBjYXNlcyB3ZXJlIGF0IHRoZSBsb3dlc3QgT+KCgiB2YWx1ZSB0aHJlZSBjb25zZWN1dGl2ZSBNT+KCgiBtZWFzdXJlcyBmdWxsIGJlbG93IG91ciBTTVIgYW5kIGZpZnRoIHBlcmNlbnRpbGUgb2YgdGhlIE1PMiB2YWx1ZXMgb2JzZXJ2ZWQgYXQgZGlzc29sdmVkIE8yIGxldmVscyA+IDgwJS4gSW4gdGhlIG1vZGVsIG91dHB1dCB0aGVzZSBhcmUgY2FsbGVkIG5iX21vMl9jb25mb3JtaW5nIHBvaW50cy4NCg0KYGBge3J9DQpvMmNyaXRfbnVtZXJpY2FsX3llcyA8LSBwY3JpdF9tb2RlbF9kZiAlPiUgDQogIGRwbHlyOjpmaWx0ZXIocGNyaXRfbmJfbW8yX2NvbmZvcm1pbmcgPiAyKSAlPiUgDQogIHB1bGwoaWQpDQoNCnBhc3RlMCgiQmFzZWQgb24gdGhpcyBydWxlIHRoZXJlIGFyZSAiLCBsZW5ndGgobzJjcml0X251bWVyaWNhbF95ZXMpLCAiIGZpc2ggd2l0aCBwb3NzaWJsZSBP4oKCY3JpdHMuIikNCmBgYA0KDQojIyMgT+KCgmNyaXQgbnVtYmVycw0KDQpXZSB3aWxsIG5vdyBjaGVjayB0byBzZWUgd2hpY2ggZmlzaCBoYXZlIE88c3ViPjJjcml0PC9zdWI+IGJhc2VkIG9uIHRoZSBudW1lcmljYWwgYW5kIHZpc3VhbCBpbnNwZWN0aW9ucy4gICANCg0KVGhlcmUgYXJlIGVpZ2h0IGZpc2ggdGhhdCBib3RoIG51bWVyaWNhbGx5IGFuZCB2aXN1YWxseSB3ZXJlIGRldGVybWluZWQgdG8gaGF2ZSBhIE88c3ViPjJjcml0PC9zdWI+LiBUd28gZmlzaCB3aXRoIGEgdmlzdWFsbHkgY29uZmlybWVkIE88c3ViPjJjcml0PC9zdWI+LCB0aGF0IGRpZCBub3QgbWVldCB0aGUgbnVtZXJpY2FsIGNyaXRlcmlhLiBFaWdodCBmaXNoIG1lZXQgdGhlIG51bWVyaWNhbCBjcml0ZXJpYSwgYnV0IGRpZCBub3QgcGFzcyB0aGUgdmlzdWFsIGluc3BlY3Rpb24uICANCg0KYGBge3J9DQpzaGFyZWRfZWxlbWVudHMgPC0gaW50ZXJzZWN0KG8yY3JpdF92aXN1YWxfeWVzLCBvMmNyaXRfbnVtZXJpY2FsX3llcykNCg0KdW5pcXVlX3RvX3Zpc3VhbCA8LSBzZXRkaWZmKG8yY3JpdF92aXN1YWxfeWVzLCBvMmNyaXRfbnVtZXJpY2FsX3llcykNCg0KdW5pcXVlX3RvX251bWVyaWNhbCA8LSBzZXRkaWZmKG8yY3JpdF9udW1lcmljYWxfeWVzLCBvMmNyaXRfdmlzdWFsX3llcykNCg0KbGlzdCgNCiAgU2hhcmVkID0gc2hhcmVkX2VsZW1lbnRzLA0KICBVbmlxdWVfdG9fdmlzdWFsID0gdW5pcXVlX3RvX3Zpc3VhbCwNCiAgVW5pcXVlX3RvX251bWVyaWNhbCA9IHVuaXF1ZV90b19udW1lcmljYWwNCikNCmBgYA0KDQojIyMjIPCfk4ggUmVzdWx0cw0KDQpMb29raW5nIGF0IG1lYW4gTzxzdWI+MmNyaXQ8L3N1Yj4gZXN0aW1hdGVzLiAgIA0KDQpgYGB7cn0NCnBvcF9tYXNzIDwtIHBjcml0X21vZGVsX21ldGEgJT4lIA0KICBkcGx5cjo6cmVmcmFtZSgNCiAgICBtZWFuX21hc3MgPSBtZWFuKG1hc3MpDQogICkgJT4lIA0KICBwdWxsKG1lYW5fbWFzcykNCg0KcGNyaXRfbW9kZWxfbWV0YSAlPiUgDQogIGRwbHlyOjpmaWx0ZXIoaWQgJWluJSBzaGFyZWRfZWxlbWVudHMpICU+JSANCiAgZHBseXI6Om11dGF0ZShyZWxfbWFzcyA9IG1hc3MvcG9wX21hc3MpICU+JSANCiAgZHBseXI6OnJlZnJhbWUoDQogICAgbl9vMmNyaXQgPSBsZW5ndGgocGNyaXRfdmF1bGUpLA0KICAgIG1lYW5fbzJjcml0ID0gbWVhbihwY3JpdF92YXVsZSksDQogICAgbWluX28yY3JpdCA9IG1pbihwY3JpdF92YXVsZSksDQogICAgbWF4X28yY3JpdCA9IG1heChwY3JpdF92YXVsZSksDQogICAgbWVhbl9tYXNzID0gbWVhbihtYXNzKSwNCiAgICBtZWFuX3JlbF9tYXNzID0gbWVhbiAocmVsX21hc3MpLA0KICAgIHNkX21hc3MgPSBzZChtYXNzKSwNCiAgICBzZF9yZWxfbWFzcyA9IHNkKHJlbF9tYXNzKQ0KICApICU+JSANCiAgZ3QoKSAlPiUgDQogICAgZm10X251bWJlcigNCiAgICBjb2x1bW5zID0gd2hlcmUoaXMubnVtZXJpYyksDQogICAgZGVjaW1hbHMgPSAyDQogICkNCmBgYA0KDQoNCmBgYHtyfQ0KbWluX2RvcyA8LSBzbG9wZV90aWR5ICU+JSANCiAgZHBseXI6OmZpbHRlcighKGlkICVpbiUgc2hhcmVkX2VsZW1lbnRzKSkgJT4lDQogIGdyb3VwX2J5KGlkKSAlPiUgDQogIGRwbHlyOjpyZWZyYW1lKA0KICAgIG1pbl9kbyA9IG1pbihETykNCiAgKQ0KDQptaW5fZG9zICU+JSANCiAgZHBseXI6OnJlZnJhbWUoDQogICAgbWVhbl9taW5fZG8gPSByb3VuZChtZWFuKG1pbl9kbyksMiksDQogICAgc2RfbWluX2RvID0gcm91bmQoc2QobWluX2RvKSwyKSwNCiAgICBtaW5fbWluX2RvID0gcm91bmQobWluKG1pbl9kbyksMikNCiAgKSAlPiUgDQogIGd0KCkNCmBgYA0KDQoNCg0KDQpgYGB7cn0NCnBjcml0X21vZGVsX21ldGEgJT4lIA0KICBkcGx5cjo6ZmlsdGVyKGlkICVpbiUgc2hhcmVkX2VsZW1lbnRzKSAlPiUgDQogIGRwbHlyOjpncm91cF9ieShzYWxpbml0eV9ncm91cCkgJT4lIA0KICBkcGx5cjo6cmVmcmFtZSgNCiAgICBuX28yY3JpdCA9IGxlbmd0aChwY3JpdF92YXVsZSksDQogICAgbWVhbl9vMmNyaXQgPSBtZWFuKHBjcml0X3ZhdWxlKSwNCiAgICBtaW5fbzJjcml0ID0gbWluKHBjcml0X3ZhdWxlKSwNCiAgICBtYXhfbzJjcml0ID0gbWF4KHBjcml0X3ZhdWxlKSwNCiAgICBtZWFuX21hc3MgPSBtZWFuKG1hc3MpDQogICkgJT4lIA0KICBndCgpDQpgYGANCg0KDQpUaGVzZSBhcmUgdGhlIDggZmlzaCB0aGF0IG1lZXQgbnVtZXJpY2FsIGNyaXRlcmlhIGFuZCBwYXNzZWQgdmlzdWFsIGluc3BlY3Rpb24uDQoNCmBgYHtyLCBmaWcuaGVpZ2h0ID0gMywgZmlnLndpZHRoPSA4fQ0KDQpwYXIobWZyb3cgPSBjKDEsIDIpLCBtYXIgPSBjKDQsIDQsIDQsIDIpKSANCg0KZm9yIChpZF9pIGluIHNoYXJlZF9lbGVtZW50cykgew0KICANCiAgY29tbWVudCA8LSBwY3JpdF9tb2RlbF9tZXRhICU+JSANCiAgICBkcGx5cjo6ZmlsdGVyKGlkID09IGlkX2kpICU+JSANCiAgICBkcGx5cjo6c2xpY2UoMSkgJT4lIA0KICAgIGRwbHlyOjptdXRhdGUoY29tbWVudCA9IGlmX2Vsc2UoaXMubmEoY29tbWVudHMpLCAiIiwgcGFzdGUwKCIjIiwgY29tbWVudHMpKSkgJT4lIA0KICAgIHB1bGwoY29tbWVudCkNCiAgDQogIHIyIDwtIHBjcml0X21vZGVsX21ldGEgJT4lIA0KICAgIGRwbHlyOjpmaWx0ZXIoaWQgPT0gaWRfaSkgJT4lIA0KICAgIGRwbHlyOjptdXRhdGUocGNyaXRfcjIgPSByb3VuZChwY3JpdF9yMiwgMykpICU+JSANCiAgICBkcGx5cjo6cHVsbChwY3JpdF9yMikNCiAgDQogIGNvbmZvcm1pbmcgPC0gcGNyaXRfbW9kZWxfbWV0YSAlPiUgDQogICAgZHBseXI6OmZpbHRlcihpZCA9PSBpZF9pKSAlPiUgDQogICAgZHBseXI6Om11dGF0ZShwY3JpdF9uYl9tbzJfY29uZm9ybWluZyA9IHJvdW5kKHBjcml0X25iX21vMl9jb25mb3JtaW5nLCAzKSkgJT4lIA0KICAgIGRwbHlyOjpwdWxsKHBjcml0X25iX21vMl9jb25mb3JtaW5nKQ0KICANCiAgUCA8LSBwY3JpdF9tb2RlbF9tZXRhICU+JSANCiAgICBkcGx5cjo6ZmlsdGVyKGlkID09IGlkX2kpICU+JSANCiAgICBkcGx5cjo6bXV0YXRlKHBjcml0X3AgPSByb3VuZChwY3JpdF9wLCAzKSkgJT4lIA0KICAgIGRwbHlyOjpwdWxsKHBjcml0X3ApDQogIA0KICBTTVIgPC0gcGNyaXRfbW9kZWxfbWV0YSAlPiUgDQogICAgZHBseXI6OmZpbHRlcihpZCA9PSBpZF9pKSAlPiUgDQogICAgZHBseXI6Om11dGF0ZShwY3JpdF9zbXIgPSByb3VuZChwY3JpdF9zbXIsIDMpKSAlPiUgDQogICAgZHBseXI6OnB1bGwocGNyaXRfc21yKQ0KICANCiAgbG93ZXN0TU8yIDwtIHBjcml0X21vZGVsX21ldGEgJT4lIA0KICAgIGRwbHlyOjpmaWx0ZXIoaWQgPT0gaWRfaSkgJT4lIA0KICAgIGRwbHlyOjptdXRhdGUocGNyaXRfbG93ZXN0TU8yID0gcm91bmQocGNyaXRfbG93ZXN0TU8yLCAzKSkgJT4lIA0KICAgIGRwbHlyOjpwdWxsKHBjcml0X2xvd2VzdE1PMikNCiAgDQogICMgR2VuZXJhdGUgYW5kIHJlbmRlciB0aGUgcGxvdA0KICBwbG90TzJjcml0KG8yY3JpdG9iaiA9IHBjcml0X21vZGVsc1tbaWRfaV1dKQ0KICANCiAgIyBBZGQgYSB0aXRsZQ0KICBtdGV4dCgNCiAgICB0ZXh0ID0gcGFzdGUwKGlkX2ksICIgIiwgY29tbWVudCksDQogICAgc2lkZSA9IDMsIGxpbmUgPSAyLCBhZGogPSAwLA0KICAgIGNvbCA9ICJibHVlIiwgZm9udCA9IDIsIGNleCA9IDEuMg0KICApDQogIA0KICBtdGV4dCgNCiAgICB0ZXh0ID0gcGFzdGUwKCJSMiA9ICIsIHIyLCAiOyBwID0gIiwgUCwgIjsgQ1AgPCBTTVIgPSAiLCBjb25mb3JtaW5nLCAiOyBTTVIgPSAiLCBTTVIsICI7IGxvd2VzdE1PMiA9ICIsbG93ZXN0TU8yKSwNCiAgICBzaWRlID0gMywgbGluZSA9IDEsIGFkaiA9IDAsDQogICAgY29sID0gImJsdWUiLCBmb250ID0gMSwgY2V4ID0gMC44DQogICkNCn0NCmBgYA0KDQoNClB1bGxpbmcgbW9kZWwgbWV0aG9kIGFuZCBsZXRoYWwgcG9pbnRzDQoNCmBgYHtyfQ0KaWRzIDwtIHBjcml0X21vZGVsX21ldGEgJT4lIA0KICBkcGx5cjo6ZGlzdGluY3QoaWQpICU+JSANCiAgZHBseXI6OnB1bGwoKQ0KDQpwY3JpdF9tZWFuX3Bsb3QgPC0gbGlzdCgpDQoNCmZvciAoaWRfaSBpbiBpZHMpIHsNCiAgYmVzdF9tb2RfdHlwZSA8LSBiZXN0X21vZCAlPiUNCiAgICBkcGx5cjo6ZmlsdGVyKGlkID09IGlkX2kpICU+JSANCiAgICBwdWxsKG1vZGVsX3R5cGUpDQogIG1vZCA8LSBwY3JpdF9tb2RlbHNbW2lkX2ldXQ0KICBkYXRhIDwtIG1vZCRvcmlnRGF0YQ0KICBtaW5fbGVhdGhhbCA8LSBtaW4obW9kJGxldGhhbFBvaW50cykNCiAgbWV0aG9kIDwtIG1vZCRNZXRob2QNCiAgcGNyaXQgPC0gbW9kJG8yY3JpdA0KICBkYXRhX3Nsb3BlIDwtIGRhdGEgJT4lDQogICAgZHBseXI6Om11dGF0ZShuX3JvdyA9IDE6bnJvdyguKSwNCiAgICAgICAgICAgICAgICAgIGxlYXRoYWwgPSBpZl9lbHNlKG5fcm93ID49IG1pbl9sZWF0aGFsLCAibm9ucmVndWxhdGlvbiIsICJyZWd1bGF0aW9uIiksDQogICAgICAgICAgICAgICAgICBtZXRob2QgPSBtZXRob2QsDQogICAgICAgICAgICAgICAgICBwY3JpdCA9IHBjcml0LA0KICAgICAgICAgICAgICAgICAgYmVzdF9tb2RfdHlwZSA9IGJlc3RfbW9kX3R5cGUNCiAgICApDQogIHBjcml0X21lYW5fcGxvdFtbaWRfaV1dIDwtIGRhdGFfc2xvcGUNCn0NCg0KcGNyaXRfbWVhbl9wbG90X2RmIDwtIGJpbmRfcm93cyhwY3JpdF9tZWFuX3Bsb3QpDQpgYGANCg0KDQojIyMjIEZpZ3VyZSBTMTENCg0KTWFraW5nIGEgTzJjcml0IHBsb3QgZm9yIGVhY2ggb2YgdGhlIGVpZ2h0IGZpc2ggdGhhdCBwYXN0IHZpc3VhbCBhbmQgIG51bWVyaWNhbCBhc3Nlc3NtZW50IG9uIHRoZSBzYW1lIGF4aXMuICAgDQoNCmBgYHtyfQ0KcGNyaXRfc3VtbWFyeSA8LSBwY3JpdF9tZWFuX3Bsb3RfZGYgJT4lDQogIGRwbHlyOjpmaWx0ZXIoaWQgJWluJSBzaGFyZWRfZWxlbWVudHMpICU+JSANCiAgZ3JvdXBfYnkoaWQpICU+JQ0KICBzdW1tYXJpc2UoDQogICAgU01SID0gZmlyc3QoU01SKSwNCiAgICBwY3JpdCA9IGZpcnN0KHBjcml0KSwNCiAgICBtYXhfRE8gPSBtYXgoRE8sIG5hLnJtID0gVFJVRSksDQogICAgbWV0aG9kID0gZmlyc3QobWV0aG9kKSwNCiAgICAuZ3JvdXBzID0gImRyb3AiDQogICkNCg0Kc21vb3RoX3NlZ21lbnRzIDwtIG1hcF9kZnIodW5pcXVlKHBjcml0X3N1bW1hcnkkaWQpLCBmdW5jdGlvbihjdXJfaWQpIHsNCiAgDQogIHN1bV9pbmZvIDwtIHBjcml0X3N1bW1hcnkgJT4lIGZpbHRlcihpZCA9PSBjdXJfaWQpDQogIFNNUl92YWwgPC0gc3VtX2luZm8kU01SDQogIG1ldGhvZF92YWwgPC0gc3VtX2luZm8kbWV0aG9kDQogIA0KICBkZiA8LSBwY3JpdF9tZWFuX3Bsb3RfZGYgJT4lDQogICAgZmlsdGVyKGlkID09IGN1cl9pZCwgbGVhdGhhbCA9PSAibm9ucmVndWxhdGlvbiIpDQogIA0KICBpZihucm93KGRmKSA8IDIpIHJldHVybihOVUxMKQ0KICANCiAgIyBGaXQgdGhlIGFwcHJvcHJpYXRlIG1vZGVsDQogIGlmKG1ldGhvZF92YWwgPT0gInRocm91Z2hfb3JpZ2luIikgew0KICAgIG1vZGVsIDwtIGxtKE1PMiB+IERPICsgMCwgZGF0YSA9IGRmKQ0KICB9IGVsc2Ugew0KICAgIG1vZGVsIDwtIGxtKE1PMiB+IERPLCBkYXRhID0gZGYpDQogIH0NCiAgDQogIGNvZWZzIDwtIGNvZWYobW9kZWwpDQogIA0KICBpZihtZXRob2RfdmFsID09ICJ0aHJvdWdoX29yaWdpbiIpIHsNCiAgICBzbG9wZSA8LSBjb2Vmc1sxXQ0KICAgIERPX3N0YXJ0IDwtIDANCiAgICBET19zdG9wIDwtIFNNUl92YWwgLyBzbG9wZQ0KICB9IGVsc2Ugew0KICAgIGludGVyY2VwdCA8LSBjb2Vmc1sxXQ0KICAgIHNsb3BlIDwtIGNvZWZzWzJdDQogICAgIyBGb3IgTFNfcmVnLCBzdGFydCBhdCAwIGlmIHByZWRpY3Rpb24gaXMgYWxyZWFkeSBwb3NpdGl2ZSwNCiAgICAjIG9yIGF0IHRoZSBETyB3aGVyZSBwcmVkaWN0ZWQgTU8yIGJlY29tZXMgMCBpZiBub3QuDQogICAgRE9fc3RhcnQgPC0gaWYoaW50ZXJjZXB0IDwgMCkgLWludGVyY2VwdCAvIHNsb3BlIGVsc2UgMA0KICAgIERPX3N0b3AgPC0gKFNNUl92YWwgLSBpbnRlcmNlcHQpIC8gc2xvcGUNCiAgfQ0KICANCiAgIyBJTVBPUlRBTlQ6IERvIG5vdCBib3VuZCBET19zdG9wIGJ5IG1heChkZiRETykgaWYgeW91IHdhbnQgdG8gZXh0cmFwb2xhdGUNCiAgIyBET19zdG9wIDwtIG1pbihET19zdG9wLCBtYXgoZGYkRE8sIG5hLnJtID0gVFJVRSkpICAjIDwtLSBSZW1vdmUgb3IgY29tbWVudCBvdXQgdGhpcyBsaW5lDQogIA0KICAjIENyZWF0ZSBhIHNlcXVlbmNlIGZyb20gRE9fc3RhcnQgdG8gRE9fc3RvcA0KICBET19zZXEgPC0gc2VxKERPX3N0YXJ0LCBET19zdG9wLCBsZW5ndGgub3V0ID0gMTAwKQ0KICANCiAgIyBDYWxjdWxhdGUgdGhlIHByZWRpY3RlZCBNTzIgdmFsdWVzIGZvciB0aGlzIHNlcXVlbmNlDQogIE1PMl9wcmVkIDwtIGlmKG1ldGhvZF92YWwgPT0gInRocm91Z2hfb3JpZ2luIikgew0KICAgIHNsb3BlICogRE9fc2VxDQogIH0gZWxzZSB7DQogICAgaW50ZXJjZXB0ICsgc2xvcGUgKiBET19zZXENCiAgfQ0KICANCiAgdGliYmxlKA0KICAgIGlkID0gY3VyX2lkLA0KICAgIERPID0gRE9fc2VxLA0KICAgIE1PMl9wcmVkID0gTU8yX3ByZWQNCiAgKSANCiAgfSkNCg0KRmlnXzQgPC0gZ2dwbG90KCkgKw0KICBnZW9tX3BvaW50KGRhdGEgPSBwY3JpdF9tZWFuX3Bsb3RfZGYgJT4lDQogICAgICAgICAgICAgICBkcGx5cjo6ZmlsdGVyKGlkICVpbiUgc2hhcmVkX2VsZW1lbnRzKSwgDQogICAgICAgICAgICAgYWVzKHggPSBETywgeSA9IE1PMiwgY29sb3VyID0gaWQsIGFscGhhID0gbGVhdGhhbCkpICsNCiAgc2NhbGVfYWxwaGFfbWFudWFsKHZhbHVlcyA9IGMoInJlZ3VsYXRpb24iID0gMC4yLCAibm9ucmVndWxhdGlvbiIgPSAwLjgpKSArDQogIGd1aWRlcyhhbHBoYSA9ICJub25lIikgKw0KICBnZW9tX2xpbmUoZGF0YSA9IHNtb290aF9zZWdtZW50cywNCiAgICAgICAgICAgIGFlcyh4ID0gRE8sIHkgPSBNTzJfcHJlZCwgY29sb3VyID0gaWQpLCBzaXplID0gMS4yKSArDQogIGdlb21fc2VnbWVudChkYXRhID0gcGNyaXRfc3VtbWFyeSwNCiAgICAgICAgICAgICBhZXMoeCA9IHBjcml0LCB4ZW5kID0gbWF4X0RPLCB5ID0gU01SLCB5ZW5kID0gU01SKSkgKw0KICBnZW9tX3NlZ21lbnQoZGF0YSA9IHBjcml0X3N1bW1hcnksDQogICAgICAgICAgICAgICBhZXMoeCA9IHBjcml0LCB4ZW5kID0gcGNyaXQsIHkgPSAwLCB5ZW5kID0gU01SKSkgKw0KICBzY2FsZV9jb2xvdXJfdmlyaWRpc19kKG9wdGlvbiA9ICJwbGFzbWEiKSArDQogIHRoZW1lX2NsYXNzaWMoKSArDQogIGxhYnMoDQogICAgeSA9IGV4cHJlc3Npb24oIm1nIE8iWzJdfiJoIl4tMSksDQogICAgeCA9ICJEaXNzb2x2ZWQgb3h5Z2VuIChETzsgJSBzYXR1cmF0aW9uKSINCiAgKSArDQogIHRoZW1lKGxlZ2VuZC5wb3NpdGlvbiA9ICJib3R0b20iKQ0KDQpGaWdfNA0KYGBgDQoNCg0KKipGaWd1cmUgUzEwKio6IE94eWdlbiBjb25zdW1wdGlvbiByYXRlICjhuYBP4oKCOyBtZ+KBu8K5IE/igoIgaOKBu8K5KSBhbmQgYW1iaWVudCBkaXNzb2x2ZWQgb3h5Z2VuIChETzsgJSBzYXR1cmF0aW9uKSBwbG90dGVkIGZvciBlYWNoIGZpc2ggdGhhdCBwYXNzZWQgYm90aCBwYXN0IHZpc3VhbCBhbmQgIG51bWVyaWNhbCBhc3Nlc3NtZW50IGZvciBPPHN1Yj4yY3JpdDwvc3ViPi4gVGhlIGhvcml6b250YWwgZGFzaGVkIGxpbmVzIHNob3cgU01SLCB0aGUgc29saWQgc2xvcGVkIGxpbmVzIHJlcHJlc2VudCB0aGUgbGluZWFyIGJlc3QgZml0IGZvciDhuYBPIHZhbHVlcyBiZWxvdyByZWd1bGF0b3J5IGZhaWx1cmUuIFRoZSB2ZXJ0aWNhbCBzb2lsZCBsaW5lcyByZXByZXNlbnQgTzxzdWI+MmNyaXQ8L3N1Yj4gKHRoZSBpbnRlcnNlY3Rpb24gb2YgdGhlIGxpbmVhciBiZXN0IGZpdCBmb3Ig4bmATyB2YWx1ZXMgYmVsb3cgcmVndWxhdG9yeSBmYWlsdXJlIGFuZCBTTVIpLg0KDQoNCkxldCdzIG1ha2Ugc29tZSBleGFtcGxlIHBsb3RzIGZvciBlYWNoIG1ham9yIHR5cGVzIG9mIE8yLU1PMiByZWxhdGlvbnNoaXBzIChpLmUuIDBedGheLSwgMV5zdF4tLCBhbmQgMl5uZF4vM15yZF4tIG9yZGVyIHBvbHlub21pYWwpLiAgIA0KDQpGaXJzdCBhbiBleGFtcGxlIG9mIGEgMHRoLW9yZGVyIHBvbHlub21pYWwgZml0IChjXzlfMjRub3ZfNCkuICAgDQoNCmBgYHtyfQ0KZmlnXzJhIDwtIGJheWVzX3JlZ19tb2RzX3ByZWRpY3Rpb25zICU+JQ0KICBkcGx5cjo6ZmlsdGVyKGlkID09ICJjXzlfMjRub3ZfNCIsIG1vZGVsX3R5cGUgPT0gImxtXzAiKSAlPiUgDQogIGdncGxvdCgpICsNCiAgZ2VvbV9wb2ludChhZXMoeCA9IERPLCB5ID0gTU8yKSwgY29sb3VyID0gImJsYWNrIiwgc2l6ZSA9IDMsIGFscGhhID0gMC41KSArDQogIGdlb21fc2VnbWVudCgNCiAgICBhZXMoeCA9IDAsIHhlbmQgPSAxMDAsIHkgPSAwLCB5ZW5kID0gU01SWzFdKSwgIyBTTVINCiAgICBsaW5ldHlwZSA9ICJkYXNoZWQiLCBzaXplID0gMSwgY29sb3VyID0gIiMwRjdBQUYiKSArDQogIGdlb21fc2VnbWVudCgNCiAgICBhZXMoeCA9IDAsIHhlbmQgPSAxMDAsIHkgPSBTTVJbMV0sIHllbmQgPSBTTVJbMV0pLCAjIGNvbmZvcm0NCiAgICBsaW5ldHlwZSA9ICJkYXNoZWQiLCBzaXplID0gMSwgY29sb3VyID0gIiNGRkE1MDAiKSArDQogIHRoZW1lX2ZldygpICsNCiAgICBsYWJzKA0KICAgIHkgPSBleHByZXNzaW9uKCJtZyBPIlsyXX4iaCJeLTEpLA0KICAgIHggPSBOVUxMDQogICkgKw0KICB0aGVtZSgNCiAgICBheGlzLnRpdGxlLnggPSBlbGVtZW50X2JsYW5rKCksDQogICAgYXhpcy50ZXh0LnggPSBlbGVtZW50X2JsYW5rKCksDQogICAgYXhpcy50aWNrcy54ID0gZWxlbWVudF9ibGFuaygpDQogICkNCmBgYA0KDQoNClNlY29uZCBhbiBleGFtcGxlIG9mIGEgMXN0LW9yZGVyIHBvbHlub21pYWwgYmVzdCBmaXQgKGRfOV8yNW5vdl8zKS4gIA0KDQpgYGB7cn0NCmZpZ18yYiA8LSBiYXllc19yZWdfbW9kc19wcmVkaWN0aW9ucyAlPiUNCiAgZHBseXI6OmZpbHRlcihpZCA9PSAiZF85XzI1bm92XzMiLCBtb2RlbF90eXBlID09ICJsbV8xIikgJT4lIA0KICBnZ3Bsb3QoKSArDQogIGdlb21fcG9pbnQoYWVzKHggPSBETywgeSA9IE1PMiksIGNvbG91ciA9ICJibGFjayIsIHNpemUgPSAzLCBhbHBoYSA9IDAuNSkgKw0KICAjIGdlb21fbGluZShhZXMoeCA9IERPLCB5ID0gcHJlZGljdGVkICogbWFzcyksIHNpemUgPSAyLCBjb2xvdXIgPSAiYmx1ZSIpICsNCiAgZ2VvbV9zZWdtZW50KA0KICAgIGFlcyh4ID0gMCwgeGVuZCA9IDEwMCwgeSA9IDAsIHllbmQgPSBTTVJbMV0pLCAjIFNNUg0KICAgIGxpbmV0eXBlID0gImRhc2hlZCIsIHNpemUgPSAxLCBjb2xvdXIgPSAiIzBGN0FBRiIpICsNCiAgZ2VvbV9zZWdtZW50KA0KICAgIGFlcyh4ID0gMCwgeGVuZCA9IDEwMCwgeSA9IFNNUlsxXSwgeWVuZCA9IFNNUlsxXSksICMgY29uZm9ybQ0KICAgIGxpbmV0eXBlID0gImRhc2hlZCIsIHNpemUgPSAxLCBjb2xvdXIgPSAiI0ZGQTUwMCIpICsNCiAgdGhlbWVfZmV3KCkgKw0KICBsYWJzKA0KICAgIHkgPSBleHByZXNzaW9uKCJtZyBPIlsyXX4iaCJeLTEpLA0KICAgIHggPSBOVUxMDQogICkgKw0KICB0aGVtZSgNCiAgICBheGlzLnRleHQueCA9IGVsZW1lbnRfYmxhbmsoKSwNCiAgICBheGlzLnRpY2tzLnggPSBlbGVtZW50X2JsYW5rKCkNCiAgKQ0KDQpgYGANCg0KDQpMYXN0IGFuIGV4YW1wbGUgb2YgYSBmaXNoIGJlc3QgZml0IHdpdGggYSBoaWdoZXIgb3JkZXIgcG9seW5vbWlhbCAoMm5kLW9yZGVyKS4NCg0KYGBge3J9DQojIERlZmluZSB0aGUgZXhhbXBsZSBJRA0KcGNyaXRfZXhhbXBsZSA8LSAiYl8wXzI0bm92XzEiDQoNCiMgMS4gUHJlcGFyZSBzdW1tYXJ5IGRhdGEgZm9yIHRoZSBzZWxlY3RlZCBJRA0KcGNyaXRfc3VtbWFyeV9leGFtcGxlIDwtIHBjcml0X21lYW5fcGxvdF9kZiAlPiUNCiAgZmlsdGVyKGlkID09IHBjcml0X2V4YW1wbGUpICU+JQ0KICBzdW1tYXJpc2UoDQogICAgU01SID0gZmlyc3QoU01SKSwNCiAgICBwY3JpdCA9IGZpcnN0KHBjcml0KSwNCiAgICBtYXhfRE8gPSBtYXgoRE8sIG5hLnJtID0gVFJVRSksDQogICAgbWV0aG9kID0gZmlyc3QobWV0aG9kKSwNCiAgICAuZ3JvdXBzID0gImRyb3AiDQogICkNCg0KIyAyLiBTdWJzZXQgdGhlIG9yaWdpbmFsIG5vbnJlZ3VsYXRpb24gZGF0YQ0KZGYgPC0gcGNyaXRfbWVhbl9wbG90X2RmICU+JQ0KICBmaWx0ZXIoaWQgPT0gcGNyaXRfZXhhbXBsZSwgbGVhdGhhbCA9PSAibm9ucmVndWxhdGlvbiIpDQoNCiMgMy4gRml0IG1vZGVsIGFuZCBnZW5lcmF0ZSBwcmVkaWN0aW9ucw0KaWYgKG5yb3coZGYpID49IDIpIHsNCiAgU01SX3ZhbCA8LSBwY3JpdF9zdW1tYXJ5X2V4YW1wbGUkU01SDQogIG1ldGhvZF92YWwgPC0gcGNyaXRfc3VtbWFyeV9leGFtcGxlJG1ldGhvZA0KICANCiAgaWYgKG1ldGhvZF92YWwgPT0gInRocm91Z2hfb3JpZ2luIikgew0KICAgIG1vZGVsIDwtIGxtKE1PMiB+IERPICsgMCwgZGF0YSA9IGRmKQ0KICAgIHNsb3BlIDwtIGNvZWYobW9kZWwpWzFdDQogICAgRE9fc3RhcnQgPC0gMA0KICAgIERPX3N0b3AgPC0gU01SX3ZhbCAvIHNsb3BlDQogICAgTU8yX3ByZWQgPC0gc2xvcGUgKiBzZXEoRE9fc3RhcnQsIERPX3N0b3AsIGxlbmd0aC5vdXQgPSAxMDApDQogIH0gZWxzZSB7DQogICAgbW9kZWwgPC0gbG0oTU8yIH4gRE8sIGRhdGEgPSBkZikNCiAgICBpbnRlcmNlcHQgPC0gY29lZihtb2RlbClbMV0NCiAgICBzbG9wZSA8LSBjb2VmKG1vZGVsKVsyXQ0KICAgIERPX3N0YXJ0IDwtIGlmIChpbnRlcmNlcHQgPCAwKSAtaW50ZXJjZXB0IC8gc2xvcGUgZWxzZSAwDQogICAgRE9fc3RvcCA8LSAoU01SX3ZhbCAtIGludGVyY2VwdCkgLyBzbG9wZQ0KICAgIERPX3NlcSA8LSBzZXEoRE9fc3RhcnQsIERPX3N0b3AsIGxlbmd0aC5vdXQgPSAxMDApDQogICAgTU8yX3ByZWQgPC0gaW50ZXJjZXB0ICsgc2xvcGUgKiBET19zZXENCiAgfQ0KICANCiAgRE9fc2VxIDwtIHNlcShET19zdGFydCwgRE9fc3RvcCwgbGVuZ3RoLm91dCA9IDEwMCkNCiAgDQogIHNtb290aF9zZWdtZW50IDwtIHRpYmJsZSgNCiAgICBpZCA9IHBjcml0X2V4YW1wbGUsDQogICAgRE8gPSBET19zZXEsDQogICAgTU8yX3ByZWQgPSBNTzJfcHJlZA0KICApDQp9DQoNCiMgNC4gUGxvdCB0aGUgZmlndXJlDQpmaWdfMmMgPC0gZ2dwbG90KCkgKw0KICBnZW9tX3BvaW50KA0KICAgIGRhdGEgPSBwY3JpdF9tZWFuX3Bsb3RfZGYgJT4lIGZpbHRlcihpZCA9PSBwY3JpdF9leGFtcGxlKSwNCiAgICBhZXMoeCA9IERPLCB5ID0gTU8yLCBjb2xvdXIgPSBsZWF0aGFsKSwgc2l6ZSA9IDMsIGFscGhhID0gMC41DQogICkgKw0KICBzY2FsZV9jb2xvdXJfbWFudWFsKHZhbHVlcyA9IGMoInJlZ3VsYXRpb24iID0gImJsYWNrIiwgIm5vbnJlZ3VsYXRpb24iID0gIiNjMzAwMTAiKSkgKw0KICANCiAgIyBnZW9tX3Ntb290aChkYXRhID0gcGNyaXRfbWVhbl9wbG90X2RmICU+JSBmaWx0ZXIoaWQgPT0gcGNyaXRfZXhhbXBsZSksDQogICMgICAgICAgICAgICAgYWVzKHggPSBETywgeSA9IE1PMiksIG1ldGhvZCA9ICJsbSIsIGZvcm11bGEgPSB5IH4gcG9seSh4LCAzKSwgc2UgPSBGQUxTRSwgYWxwaGEgPSAwLjgsIHNpemUgPSAyLCBmdWxscmFuZ2UgPSBUUlVFKSArDQogIA0KICBnZW9tX2xpbmUoDQogICAgZGF0YSA9IHNtb290aF9zZWdtZW50LA0KICAgIGFlcyh4ID0gRE8sIHkgPSBNTzJfcHJlZCksIHNpemUgPSAyLCBjb2xvdXIgPSAiI2MzMDAxMCIgI2ZhaWx1cmUgIHNsb3BlDQogICkgKw0KICANCiAgDQogIGdlb21fc2VnbWVudCgNCiAgICBkYXRhID0gcGNyaXRfc3VtbWFyeV9leGFtcGxlLA0KICAgIGFlcyh4ID0gMCwgeGVuZCA9IG1heF9ETywgeSA9IFNNUiwgeWVuZCA9IFNNUiksICNTTVINCiAgICBjb2xvdXIgPSAiI0ZGQTUwMCIsIGxpbmV0eXBlID0gImRhc2hlZCIsIHNpemUgPSAxKSArDQogIA0KICBnZW9tX3NlZ21lbnQoDQogICAgZGF0YSA9IHBjcml0X3N1bW1hcnlfZXhhbXBsZSwNCiAgICBhZXMoeCA9IDAsIHhlbmQgPSBtYXhfRE8sIHkgPSAwLCB5ZW5kID0gU01SKSwgI2NvbmZyb20NCiAgICBsaW5ldHlwZSA9ICJkYXNoZWQiLCBzaXplID0gMSwgY29sb3VyID0gIiMwRjdBQUYiKSArDQogIA0KICBnZW9tX3NlZ21lbnQoDQogICAgZGF0YSA9IHBjcml0X3N1bW1hcnlfZXhhbXBsZSwNCiAgICBhZXMoeCA9IHBjcml0LCB4ZW5kID0gcGNyaXQsIHkgPSAwLCB5ZW5kID0gU01SKSwgY29sb3VyID0gImRhcmtncmVlbiIsIHNpemUgPSAyICNQY3JpdA0KICApICsNCiAgDQogICNjb29yZF9jYXJ0ZXNpYW4oeWxpbSA9IGMoMCwgMC4xNikpICsNCiAgdGhlbWVfZmV3KCkgKw0KICAgbGFicygNCiAgICB5ID0gZXhwcmVzc2lvbigibWcgTyJbMl1+ImgiXi0xKSwNCiAgICB4ID0gIkRpc3NvbHZlZCBveHlnZW4gKERPOyAlIHNhdHVyYXRpb24pIg0KICApICsNCiAgdGhlbWUobGVnZW5kLnBvc2l0aW9uID0gIm5vbmUiKQ0KDQpgYGANCg0KIyMjIyBGaWd1cmUgUzEyIChGaWd1cmUgMikNCg0KSGVyZSB3ZSBhcmUgc2hvd2luZyB0aGUgb3V0IHB1dCBvZiB0aGUgQ2hhYm90IE88c3ViPjJjcml0PC9zdWI+IG1vZGVsLCANCg0KDQpgYGB7ciwgZmlnLmhlaWdodD0xMCwgZmlnLndpZHRoPTV9DQpmaWdfMl9iaW5kIDwtIGdyaWQuYXJyYW5nZShmaWdfMmEsIGZpZ18yYiwgZmlnXzJjLCBuY29sID0gMSkgI2JpbmQgcGxvdA0KYGBgDQoNCioqRmlndXJlIDIqKjogRmlndXJlIDIuIFJlcHJlc2VudGF0aXZlIGV4YW1wbGVzIG9mIGZpc2ggdGhhdCB3ZXJlIGJlc3QgbW9kZWxsZWQgYnkgYSAwdGgtb3JkZXIgcG9seW5vbWlhbCAoQSksIDFzdC1vcmRlciBwb2x5bm9taWFsIChCKSwgYW5kIDJuZC1vcmRlciBwb2x5bm9taWFsIChDKS4gVGhlIG9yYW5nZSBkYXNoZWQgbGluZSBzaG93cyB0aGUgU01SLCB3aGlsZSB0aGUgYmx1ZSBkYXNoZWQgbGluZSBzaG93cyB0aGUgdGhlb3JldGljYWwgb3h5Y29uZnJvbWluZyByZWdyZXNzaW9uLiBGb3IgcGxvdCAoQyksIHRoZSByZWQgc2xvcGVkIGxpbmUgc2hvd3MgdGhlIE8yIHJlZ3VsYXRpb24gZmFpbHVyZSByZWdyZXNzaW9uIChiYXNlZCBvbiB0aGUgcnVsZS1iYXNlZCBsaW5lYXIgcmVncmVzc2lvbiBtZXRob2Qgb2YgQ2xhaXJlYXV4IGFuZCBDaGFib3QsIDIwMTYpLCBhbmQgdGhlIHNvbGlkIGhvcml6b250YWwgZ3JlZW4gbGluZSBzaG93cyB0aGUgTzJjcml0IGVzdGltYXRlIChpbnRlcnNlY3Rpb24gb2YgTzIgZmFpbHVyZSByZWdyZXNzaW9uIGFuZCBTTVIpLiAgDQoNCmBgYHtyfQ0KIyBnZ3NhdmUoZmlsZW5hbWUgPSBwYXN0ZTAob3V0cHV0X2ZpZ193ZCwgIi4vbXNfZmlndXJlcy4vZmlndXJlLTIucGRmIiksDQojICAgICAgICBwbG90ID0gZmlnXzJfYmluZCwNCiMgICAgICAgIHdpZHRoID0gMjEwLA0KIyAgICAgICAgaGVpZ2h0ID0gMjk3LA0KIyAgICAgICAgdW5pdHMgPSAibW0iKQ0KYGBgDQoNCg0KIyBDb21hcHJpbmcgdG8gcGFzdCBkYXRhDQoNCkhlcmUsIHdlIGhhdmUgcmVjcmVhdGVkIGEgc2ltaWxhciBwbG90IHRvIHRoYXQgcHJlc2VudGVkIGluIFVyYmluYSwgR2xvdmVyLCBhbmQgRm9yc3RlciAoMjAxMileWzFdXiBhbmQgaGF2ZSBleHRyYWN0ZWQgdGhlIG1lYW4gbGV2ZWwgZGF0YSBmcm9tIEZpZ3VyZSAxYSB1c2luZyB0aGUgbWV0YURpZ2l0aXNlIHBhY2thZ2UgaW4gUl5bM11eLiBUaGlzIGRhdGEgaXMgY2FsbGVkIHVyYmluYV9ldF9hbF8yMDEyLiBUaGlzIGFsbG93cyB1cyB0byBjb21wYXJlIHRoZSBkaWZmZXJlbmNlcyBpbiBNbzIgYW5kIHRoZSByZWxhdGlvbnNoaXAgYmV0d2VlbiBNbzIgYW5kIE88c3ViPjI8L3N1Yj4uDQoNCkZpcnN0IG1ha2luZyBhIGJpbm5lZCBkYXRhIGZyYW1lIHRvIG1hdGNoIFVyYmluYSwgR2xvdmVyLCBhbmQgRm9yc3RlciAoMjAxMikgYXMgY2xvc2VseSBhcyBwb3NzaWJsZS4NCg0KYGBge3J9DQptaW5fbzJfa3BhIDwtIG1pbihzbG9wZV90aWR5JG8yX2twYSwgbmEucm0gPSBUUlVFKQ0KbWF4X28yX2twYSA8LSBtYXgoc2xvcGVfdGlkeSRvMl9rcGEsIG5hLnJtID0gVFJVRSkNCg0KdGltZV9iaW5fZGYgPC0gc2xvcGVfdGlkeSAlPiUNCiAgbXV0YXRlKG8yX2dyb3VwID0gY3V0KG8yX2twYSwgDQogICAgICAgICAgICAgICAgICAgICAgICBicmVha3MgPSBzZXEobWluX28yX2twYSwgbWF4X28yX2twYSwgbGVuZ3RoLm91dCA9IDEzKSwgIyAxMSBpbnRlcnZhbHMsIHNvIDEyIGJyZWFrcG9pbnRzDQogICAgICAgICAgICAgICAgICAgICAgICBsYWJlbHMgPSBwYXN0ZTAoIkdyb3VwICIsIDE6MTIpLCANCiAgICAgICAgICAgICAgICAgICAgICAgIGluY2x1ZGUubG93ZXN0ID0gVFJVRSkpICU+JSANCiAgZHBseXI6Omdyb3VwX2J5KG8yX2dyb3VwKSAlPiUgDQogIGRwbHlyOjpyZWZyYW1lKG1lYW5fTU8yX2cgPSBtZWFuKE1PMl9nKSozMS4yNSwNCiAgICAgICAgICAgICAgICAgbWVhbl9vMl9rcGEgPSBtZWFuKG8yX2twYSksDQogICAgICAgICAgICAgICAgIG4gPSBsZW5ndGgoTU8yX2cpKjMxLjI1LA0KICAgICAgICAgICAgICAgICBNTzJfZ19zZCA9IHNkKE1PMl9nKSozMS4yNSwNCiAgICAgICAgICAgICAgICAgbzJfa3BhX3NkID0gc2QobzJfa3BhKSkNCmBgYA0KDQojIyBGaWd1cmUgUzEzIChGaWd1cmUgMykNCg0KTm93IHRoZSBwbG90IHdpdGggb3VyIG1lYW4gZGF0YSBhbmQgdGhlIG1lYW4gZGF0YSBmcm9tIFVyYmluYSwgR2xvdmVyLCBhbmQgRm9yc3RlciAoMjAxMileWzFdXiAgIA0KDQpgYGB7cn0NCm4gPC0gc2xvcGVfdGlkeSAlPiUgDQogIGRwbHlyOjpkaXN0aW5jdChpZCkgJT4lIA0KICBucm93KC4pDQoNCiMgRXhpc3RpbmcgcGxvdA0KZmlnMyA8LSB0aW1lX2Jpbl9kZiAlPiUgDQogIGdncGxvdCgpICsNCiAgZ2VvbV9wb2ludChkYXRhID0gc2xvcGVfdGlkeSwgDQogICAgICAgICAgICAgYWVzKHkgPSBNTzJfZyAqIDMxLjI1LCB4ID0gbzJfa3BhKSwgDQogICAgICAgICAgICAgc2l6ZSA9IDIsIGNvbG9yID0gImdyZXkiLCBhbHBoYSA9IDAuMykgKyAgDQogIGdlb21fcG9pbnQoZGF0YSA9IHRpbWVfYmluX2RmLCANCiAgICAgICAgICAgICBhZXMoeSA9IG1lYW5fTU8yX2csIHggPSBtZWFuX28yX2twYSksDQogICAgICAgICAgICAgc2l6ZSA9IDMsIGNvbG91ciA9ICIjMEU0QzkyIiwgc2hvdy5sZWdlbmQgPSBGQUxTRSkgKw0KICBnZW9tX2Vycm9yYmFyKGRhdGEgPSB0aW1lX2Jpbl9kZiwNCiAgICAgICAgICAgICAgICBhZXMoeSA9IG1lYW5fTU8yX2csIHggPSBtZWFuX28yX2twYSwNCiAgICAgICAgICAgICAgICAgICAgeW1pbiA9IG1lYW5fTU8yX2cgLSBNTzJfZ19zZCwgeW1heCA9IG1lYW5fTU8yX2cgKyBNTzJfZ19zZCksIA0KICAgICAgICAgICAgICAgIHdpZHRoID0gMC4xNSwgY29sb3VyID0gIiMwRTRDOTIiKSArDQogIGdlb21fZXJyb3JiYXJoKGRhdGEgPSB0aW1lX2Jpbl9kZiwgDQogICAgICAgICAgICAgICAgIGFlcyh5ID0gbWVhbl9NTzJfZywgeCA9IG1lYW5fbzJfa3BhLA0KICAgICAgICAgICAgICAgICAgICAgeG1pbiA9IG1lYW5fbzJfa3BhIC0gbzJfa3BhX3NkLCB4bWF4ID0gbWVhbl9vMl9rcGEgKyBvMl9rcGFfc2QpLCANCiAgICAgICAgICAgICAgICAgaGVpZ2h0ID0gMC40LCBjb2xvdXIgPSAiIzBFNEM5MiIpICsNCiAgZ2VvbV9wb2ludChkYXRhID0gdXJiaW5hX2V0X2FsXzIwMTIsIA0KICAgICAgICAgICAgIGFlcyh4ID0gbzJfbWVhbiwgeSA9IG1vMl9tZWFuKSwgDQogICAgICAgICAgICAgc2l6ZSA9IDMsIHNoYXBlID0gMSwgZmlsbCA9ICIjRDIxRjNDIiwgY29sb3IgPSAiI0QyMUYzQyIsIHN0cm9rZSA9IDEpICsNCiAgZ2VvbV9lcnJvcmJhcihkYXRhID0gdXJiaW5hX2V0X2FsXzIwMTIsIA0KICAgICAgICAgICAgICAgIGFlcyh4ID0gbzJfbWVhbiwgeW1pbiA9IG1vMl9tZWFuIC0gbW8yX3NkLCB5bWF4ID0gbW8yX21lYW4gKyBtbzJfc2QpLCANCiAgICAgICAgICAgICAgICB3aWR0aCA9IDAuMiwgY29sb3VyID0gIiNEMjFGM0MiKSArDQogIGdlb21fZXJyb3JiYXJoKGRhdGEgPSB1cmJpbmFfZXRfYWxfMjAxMiwgDQogICAgICAgICAgICAgICAgIGFlcyh5ID0gbW8yX21lYW4sIHhtaW4gPSBvMl9tZWFuIC0gbzJfc2QsIHhtYXggPSBvMl9tZWFuICsgbzJfc2QpLCANCiAgICAgICAgICAgICAgICAgaGVpZ2h0ID0gMC40LCBjb2xvdXIgPSAiI0QyMUYzQyIpICsNCiAgYW5ub3RhdGUoInRleHQiLCB4ID0gMCwgDQogICAgICAgICAgIHkgPSAxNiwgDQogICAgICAgICAgIGxhYmVsID0gYnF1b3RlKGF0b3AoIkN1cnJlbnQgZGF0YSAoYmx1ZSksICIgKiBpdGFsaWMobikgKiAiID0gIiAqIC4obiksDQogICAgICAgICAgICAgICAgICAgICAgICAiVXJiaW5hIGRhdGEgKHJlZCksICIgKiBpdGFsaWMobikgKiAiID0gNjciKSksICANCiAgICAgICAgIGhqdXN0ID0gMCwgdmp1c3QgPSAxLCBzaXplID0gNCkgKw0KICB0aGVtZSgNCiAgICBsZWdlbmQucG9zaXRpb24gPSAibm9uZSIsDQogICAgcGFuZWwuZ3JpZC5tYWpvciA9IGVsZW1lbnRfYmxhbmsoKSwNCiAgICBwYW5lbC5ncmlkLm1pbm9yID0gZWxlbWVudF9ibGFuaygpLA0KICAgIHBhbmVsLmJvcmRlciA9IGVsZW1lbnRfcmVjdChjb2xvdXIgPSAiYmxhY2siLCBmaWxsID0gTkEsIGxpbmV3aWR0aCA9IDEpLA0KICAgIHBhbmVsLmJhY2tncm91bmQgPSBlbGVtZW50X3JlY3QoZmlsbCA9ICJ3aGl0ZSIsIGNvbG91ciA9IE5BKSwNCiAgICBwbG90LmJhY2tncm91bmQgPSBlbGVtZW50X3JlY3QoZmlsbCA9ICJ3aGl0ZSIsIGNvbG91ciA9IE5BKQ0KICApICsNCiAgbGFicygNCiAgc3VidGl0bGUgPSBOVUxMLA0KICB4ID0gZXhwcmVzc2lvbigiUE8iWzJdfiIoa1BhKSIpLA0KICB5ID0gZXhwcmVzc2lvbigiTU8iWzJdfiIozrxtb2wgTyJbMl1+ImciXi0xfiJoIl4tMSkpICsNCiAgc2NhbGVfeV9jb250aW51b3VzKGxpbWl0cyA9IGMoMCwgMTYpLCBicmVha3MgPSBzZXEoMCwgMTYsIGJ5ID0gMikpICsNCiAgc2NhbGVfeF9jb250aW51b3VzKGxpbWl0cyA9IGMoMCwgMjIpLCBicmVha3MgPSBzZXEoMCwgMjIsIGJ5ID0gMikpICANCg0KZmlnMw0KYGBgDQoNCioqRmlndXJlIDM6KiogTWVhbiBhbmQgc3RhbmRhcmQgZXJyb3Igb2YgbWV0YWJvbGljIHJhdGUgKE1PPHN1Yj4yPC9zdWI+IM68bW9sIE88c3ViPjI8L3N1Yj4gZ14tMV4gaF4tMV4pIGFuZCBveHlnZW4gY29uY2VudHJhdGlvbiAoUE88c3ViPjI8L3N1Yj4ga1BhKSB1c2luZyAxMiBldmVubHkgc3BhY2VkIGJpbnMgb3ZlciB0aGUgTzxzdWI+Mjwvc3ViPiByYW5nZSBvZiBvYnNlcnZlZCBkYXRhIChibHVlIGZpbGxlZCBkb3RzKS4gQ29tcGFyZWQgYWdhaW5zdCB0aGUgbWVhbiBhbmQgc3RhbmRhcmQgZXJyb3IgcmVwb3J0ZWQgaW4gVXJiaW5hICgyMDEyKV5bMV1eIChyZWQgb3BlbiBkb3RzKS4gVGhlIGdyZXkgZG90cyBhcmUgdGhlIHJhdyBvYnNlcnZlZCBkYXRhIGZvcm0gdGhlIHByZXNlbnQgc3R1ZHkuDQoNCmBgYHtyfQ0KIyBnZ3NhdmUoZmlsZW5hbWUgPSBwYXN0ZTAob3V0cHV0X2ZpZ193ZCwgIi4vbXNfZmlndXJlcy4vZmlndXJlLTMucGRmIiksDQojICAgICAgICBwbG90ID0gZmlnMywNCiMgICAgICAgIHdpZHRoID0gMjEwLA0KIyAgICAgICAgaGVpZ2h0ID0gMjk3LzIsICAjIFNwZWNpZnkgdGhlIGhlaWdodCBvZiB0aGUgcGxvdA0KIyAgICAgICAgdW5pdHMgPSAibW0iKQ0KYGBgDQoNCjwhLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLT4NCg0KIyDwn5OaIFJlZmVyZW5jZXMNCg0KPCEtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tPg0KDQpeWzFdXiBDbGFpcmVhdXgsIEcuIGFuZCBDaGFib3QsIEQuICgyMDE2KSBSZXNwb25zZXMgYnkgZmlzaGVzIHRvIGVudmlyb25tZW50YWwgaHlwb3hpYTogaW50ZWdyYXRpb24gdGhyb3VnaCBGcnkncyBjb25jZXB0IG9mIGFlcm9iaWMgbWV0YWJvbGljIHNjb3BlLiBKb3VybmFsIG9mIEZpc2ggQmlvbG9neSA8aHR0cHM6Ly9kb2kub3JnLzEwLjExMTEvamZiLjEyODMzPg0KDQpeWzJdXiBVcmJpbmEgTUEsIEdsb3ZlciBDTiwgYW5kIEZvcnN0ZXIgTUUsICgyMDEyKSBBIG5vdmVsIG94eWNvbmZvcm1pbmcgcmVzcG9uc2UgaW4gdGhlIGZyZXNod2F0ZXIgZmlzaCAqR2FsYXhpYXMgbWFjdWxhdHVzKi4gQ29tcGFyYXRpdmUgQmlvY2hlbWlzdHJ5IGFuZCBQaHlzaW9sb2d5IFBhcnQgQTogTW9sZWN1bGFyICYgSW50ZWdyYXRpdmUgUGh5c2lvbG9neS4gPGh0dHBzOi8vZG9pLm9yZy8xMC4xMDE2L2ouY2JwYS4yMDExLjExLjAxMT4NCg0KXlszXV4gUGljayBKTCwgTmFrYWdhd2EgUywgYW5kIE5vYmxlIERXQSAoMjAxOCkgUmVwcm9kdWNpYmxlLCBmbGV4aWJsZSBhbmQgaGlnaC10aHJvdWdocHV0IGRhdGEgZXh0cmFjdGlvbiBmcm9tIHByaW1hcnkgbGl0ZXJhdHVyZTogVGhlIG1ldGFEaWdpdGlzZSByIHBhY2thZ2UuIDxodHRwczovL2RvaS5vcmcvMTAuMTExMS8yMDQxLTIxMFguMTMxMTg+DQoNCjwhLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLT4NCg0KIyDwn5K7IFNlc3Npb24gaW5mb3JtYXRpb24NCg0KPCEtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tPg0KDQpgYGB7cn0NCmNpdGF0aW9uKCkNCmBgYA0KDQpIZXJlIGlzIGEgZGV0YWlsZWQgbGlzdCBvZiB0aGUgc2Vzc2lvbiBpbmZvcm1hdGlvbg0KDQpgYGB7cn0NCnNlc3Npb25pbmZvOjpzZXNzaW9uX2luZm8oKQ0KYGBgDQo=